mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-24 14:05:17 +00:00
c28bc03248
* comments from review Co-authored-by: Sunny Aggarwal <sunnya97@protonmail.ch> Co-authored-by: jmahess <maheswaran@google.com> Co-authored-by: Alexander Bezobchuk <alexanderbez@users.noreply.github.com> * add vote methods * add draft new param change permission * add and update tests * rename ParamChangePermission * account for perms becoming invalid at a later time * add debtParam to permission * add bep3 AssetParam to permissions * add pricefeed Markets to permission * add upgrade permission * move proposal passing to the begin blocker * fix iteration bug Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * address todos and audit comments * add proposal examples * refactor handler to be easier to read * address review comments * update comments Co-authored-by: Kevin Davis <kjydavis3@gmail.com> Co-authored-by: Sunny Aggarwal <sunnya97@protonmail.ch> Co-authored-by: jmahess <maheswaran@google.com> Co-authored-by: Alexander Bezobchuk <alexanderbez@users.noreply.github.com> Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com>
231 lines
8.0 KiB
Go
231 lines
8.0 KiB
Go
package committee_test
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/cosmos/cosmos-sdk/x/gov"
|
|
"github.com/cosmos/cosmos-sdk/x/params"
|
|
"github.com/cosmos/cosmos-sdk/x/upgrade"
|
|
|
|
abci "github.com/tendermint/tendermint/abci/types"
|
|
|
|
"github.com/kava-labs/kava/app"
|
|
"github.com/kava-labs/kava/x/cdp"
|
|
cdptypes "github.com/kava-labs/kava/x/cdp/types"
|
|
"github.com/kava-labs/kava/x/committee"
|
|
)
|
|
|
|
type ModuleTestSuite struct {
|
|
suite.Suite
|
|
|
|
keeper committee.Keeper
|
|
app app.TestApp
|
|
ctx sdk.Context
|
|
|
|
addresses []sdk.AccAddress
|
|
}
|
|
|
|
func (suite *ModuleTestSuite) SetupTest() {
|
|
suite.app = app.NewTestApp()
|
|
suite.keeper = suite.app.GetCommitteeKeeper()
|
|
suite.ctx = suite.app.NewContext(true, abci.Header{})
|
|
_, suite.addresses = app.GeneratePrivKeyAddressPairs(5)
|
|
}
|
|
|
|
func (suite *ModuleTestSuite) TestBeginBlock_ClosesExpired() {
|
|
suite.app.InitializeFromGenesisStates()
|
|
|
|
normalCom := committee.Committee{
|
|
ID: 12,
|
|
Members: suite.addresses[:2],
|
|
Permissions: []committee.Permission{committee.GodPermission{}},
|
|
VoteThreshold: d("0.8"),
|
|
ProposalDuration: time.Hour * 24 * 7,
|
|
}
|
|
suite.keeper.SetCommittee(suite.ctx, normalCom)
|
|
|
|
pprop1 := gov.NewTextProposal("Title 1", "A description of this proposal.")
|
|
id1, err := suite.keeper.SubmitProposal(suite.ctx, normalCom.Members[0], normalCom.ID, pprop1)
|
|
suite.NoError(err)
|
|
|
|
oneHrLaterCtx := suite.ctx.WithBlockTime(suite.ctx.BlockTime().Add(time.Hour))
|
|
pprop2 := gov.NewTextProposal("Title 2", "A description of this proposal.")
|
|
id2, err := suite.keeper.SubmitProposal(oneHrLaterCtx, normalCom.Members[0], normalCom.ID, pprop2)
|
|
suite.NoError(err)
|
|
|
|
// Run BeginBlocker
|
|
proposalDurationLaterCtx := suite.ctx.WithBlockTime(suite.ctx.BlockTime().Add(normalCom.ProposalDuration))
|
|
suite.NotPanics(func() {
|
|
committee.BeginBlocker(proposalDurationLaterCtx, abci.RequestBeginBlock{}, suite.keeper)
|
|
})
|
|
|
|
// Check expired proposals are gone
|
|
_, found := suite.keeper.GetProposal(suite.ctx, id1)
|
|
suite.False(found, "expected expired proposal to be closed")
|
|
_, found = suite.keeper.GetProposal(suite.ctx, id2)
|
|
suite.True(found, "expected non expired proposal to be not closed")
|
|
}
|
|
|
|
func (suite *ModuleTestSuite) TestBeginBlock_EnactsPassed() {
|
|
suite.app.InitializeFromGenesisStates()
|
|
|
|
// setup committee
|
|
normalCom := committee.Committee{
|
|
ID: 12,
|
|
Members: suite.addresses[:2],
|
|
Permissions: []committee.Permission{committee.GodPermission{}},
|
|
VoteThreshold: d("0.8"),
|
|
ProposalDuration: time.Hour * 24 * 7,
|
|
}
|
|
suite.keeper.SetCommittee(suite.ctx, normalCom)
|
|
|
|
// setup 2 proposals
|
|
previousCDPDebtThreshold := suite.app.GetCDPKeeper().GetParams(suite.ctx).DebtAuctionThreshold
|
|
newDebtThreshold := previousCDPDebtThreshold.Add(i(1000000))
|
|
evenNewerDebtThreshold := newDebtThreshold.Add(i(1000000))
|
|
|
|
pprop1 := params.NewParameterChangeProposal("Title 1", "A description of this proposal.",
|
|
[]params.ParamChange{{
|
|
Subspace: cdptypes.ModuleName,
|
|
Key: string(cdp.KeyDebtThreshold),
|
|
Value: string(cdp.ModuleCdc.MustMarshalJSON(newDebtThreshold)),
|
|
}},
|
|
)
|
|
id1, err := suite.keeper.SubmitProposal(suite.ctx, normalCom.Members[0], normalCom.ID, pprop1)
|
|
suite.NoError(err)
|
|
|
|
pprop2 := params.NewParameterChangeProposal("Title 2", "A description of this proposal.",
|
|
[]params.ParamChange{{
|
|
Subspace: cdptypes.ModuleName,
|
|
Key: string(cdp.KeyDebtThreshold),
|
|
Value: string(cdp.ModuleCdc.MustMarshalJSON(evenNewerDebtThreshold)),
|
|
}},
|
|
)
|
|
id2, err := suite.keeper.SubmitProposal(suite.ctx, normalCom.Members[0], normalCom.ID, pprop2)
|
|
suite.NoError(err)
|
|
|
|
// add enough votes to make the first proposal pass, but not the second
|
|
suite.NoError(suite.keeper.AddVote(suite.ctx, id1, suite.addresses[0]))
|
|
suite.NoError(suite.keeper.AddVote(suite.ctx, id1, suite.addresses[1]))
|
|
suite.NoError(suite.keeper.AddVote(suite.ctx, id2, suite.addresses[0]))
|
|
|
|
// Run BeginBlocker
|
|
suite.NotPanics(func() {
|
|
committee.BeginBlocker(suite.ctx, abci.RequestBeginBlock{}, suite.keeper)
|
|
})
|
|
|
|
// Check the param has been updated
|
|
suite.Equal(newDebtThreshold, suite.app.GetCDPKeeper().GetParams(suite.ctx).DebtAuctionThreshold)
|
|
// Check the passed proposal has gone
|
|
_, found := suite.keeper.GetProposal(suite.ctx, id1)
|
|
suite.False(found, "expected passed proposal to be enacted and closed")
|
|
_, found = suite.keeper.GetProposal(suite.ctx, id2)
|
|
suite.True(found, "expected non passed proposal to be not closed")
|
|
}
|
|
|
|
func (suite *ModuleTestSuite) TestBeginBlock_DoesntEnactFailed() {
|
|
suite.app.InitializeFromGenesisStates()
|
|
|
|
// setup committee
|
|
normalCom := committee.Committee{
|
|
ID: 12,
|
|
Members: suite.addresses[:1],
|
|
Permissions: []committee.Permission{committee.SoftwareUpgradePermission{}},
|
|
VoteThreshold: d("1.0"),
|
|
ProposalDuration: time.Hour * 24 * 7,
|
|
}
|
|
firstBlockTime := time.Date(1998, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
ctx := suite.ctx.WithBlockTime(firstBlockTime)
|
|
suite.keeper.SetCommittee(ctx, normalCom)
|
|
|
|
// setup an upgrade proposal
|
|
pprop1 := upgrade.NewSoftwareUpgradeProposal("Title 1", "A description of this proposal.",
|
|
upgrade.Plan{
|
|
Name: "upgrade-version-v0.23.1",
|
|
Time: firstBlockTime.Add(time.Second * 5),
|
|
Info: "some information about the upgrade",
|
|
},
|
|
)
|
|
id1, err := suite.keeper.SubmitProposal(ctx, normalCom.Members[0], normalCom.ID, pprop1)
|
|
suite.NoError(err)
|
|
|
|
// add enough votes to make the proposal pass
|
|
suite.NoError(suite.keeper.AddVote(ctx, id1, suite.addresses[0]))
|
|
|
|
// Run BeginBlocker 10 seconds later (5 seconds after upgrade expires)
|
|
tenSecLaterCtx := ctx.WithBlockTime(ctx.BlockTime().Add(time.Second * 10))
|
|
suite.NotPanics(func() {
|
|
suite.app.BeginBlocker(tenSecLaterCtx, abci.RequestBeginBlock{})
|
|
})
|
|
|
|
// Check the plan has not been stored
|
|
_, found := suite.app.GetUpgradeKeeper().GetUpgradePlan(tenSecLaterCtx)
|
|
suite.False(found)
|
|
// Check the passed proposal has gone
|
|
_, found = suite.keeper.GetProposal(tenSecLaterCtx, id1)
|
|
suite.False(found, "expected failed proposal to be not enacted and closed")
|
|
|
|
// Check the chain doesn't halt
|
|
oneMinLaterCtx := ctx.WithBlockTime(ctx.BlockTime().Add(time.Minute).Add(time.Second))
|
|
suite.NotPanics(func() {
|
|
suite.app.BeginBlocker(oneMinLaterCtx, abci.RequestBeginBlock{})
|
|
})
|
|
}
|
|
|
|
func (suite *ModuleTestSuite) TestBeginBlock_EnactsPassedUpgrade() {
|
|
suite.app.InitializeFromGenesisStates()
|
|
|
|
// setup committee
|
|
normalCom := committee.Committee{
|
|
ID: 12,
|
|
Members: suite.addresses[:1],
|
|
Permissions: []committee.Permission{committee.SoftwareUpgradePermission{}},
|
|
VoteThreshold: d("1.0"),
|
|
ProposalDuration: time.Hour * 24 * 7,
|
|
}
|
|
firstBlockTime := time.Date(1998, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
ctx := suite.ctx.WithBlockTime(firstBlockTime)
|
|
suite.keeper.SetCommittee(ctx, normalCom)
|
|
|
|
// setup an upgrade proposal
|
|
pprop1 := upgrade.NewSoftwareUpgradeProposal("Title 1", "A description of this proposal.",
|
|
upgrade.Plan{
|
|
Name: "upgrade-version-v0.23.1",
|
|
Time: firstBlockTime.Add(time.Minute * 1),
|
|
Info: "some information about the upgrade",
|
|
},
|
|
)
|
|
id1, err := suite.keeper.SubmitProposal(ctx, normalCom.Members[0], normalCom.ID, pprop1)
|
|
suite.NoError(err)
|
|
|
|
// add enough votes to make the proposal pass
|
|
suite.NoError(suite.keeper.AddVote(ctx, id1, suite.addresses[0]))
|
|
|
|
// Run BeginBlocker
|
|
fiveSecLaterCtx := ctx.WithBlockTime(ctx.BlockTime().Add(time.Second * 5))
|
|
suite.NotPanics(func() {
|
|
suite.app.BeginBlocker(fiveSecLaterCtx, abci.RequestBeginBlock{})
|
|
})
|
|
|
|
// Check the plan has been stored
|
|
_, found := suite.app.GetUpgradeKeeper().GetUpgradePlan(fiveSecLaterCtx)
|
|
suite.True(found)
|
|
// Check the passed proposal has gone
|
|
_, found = suite.keeper.GetProposal(fiveSecLaterCtx, id1)
|
|
suite.False(found, "expected passed proposal to be enacted and closed")
|
|
|
|
// Check the chain halts
|
|
oneMinLaterCtx := ctx.WithBlockTime(ctx.BlockTime().Add(time.Minute))
|
|
suite.Panics(func() {
|
|
suite.app.BeginBlocker(oneMinLaterCtx, abci.RequestBeginBlock{})
|
|
})
|
|
}
|
|
|
|
func TestModuleTestSuite(t *testing.T) {
|
|
suite.Run(t, new(ModuleTestSuite))
|
|
}
|