mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-26 08:15:19 +00:00
add main keeper methods
This commit is contained in:
parent
8c64fd3750
commit
f9dab88c16
@ -25,49 +25,51 @@ func NewKeeper(cdc *codec.Codec, storeKey sdk.StoreKey) Keeper {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO keeper methods - very similar to gov
|
func (k Keeper) SubmitProposal(ctx sdk.Context, proposer sdk.AccAddress, proposal types.Proposal) (uint64, sdk.Error) {
|
||||||
|
// Limit proposals to only be submitted by committee members
|
||||||
- SubmitProposal validate and store a proposal, additionally setting things like timeout
|
com, found := k.GetCommittee(ctx, proposal.CommitteeID)
|
||||||
- GetProposal
|
if !found {
|
||||||
- SetProposal
|
return 0, sdk.ErrInternal("committee doesn't exist")
|
||||||
|
|
||||||
- AddVote - add a vote to a particular proposal from a member
|
|
||||||
- GetVote
|
|
||||||
- SetVote
|
|
||||||
|
|
||||||
- GetCommittee
|
|
||||||
- SetCommittee
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
func (k Keeper) SubmitProposal(ctx sdk.Context, proposal types.Proposal) sdk.Error {
|
|
||||||
// TODO Limit proposals to only be submitted by group members
|
|
||||||
|
|
||||||
// Check group has permissions to enact proposal. As long as one permission allows the proposal then it goes through. Its the OR of all permissions.
|
|
||||||
committee, _ := k.GetCommittee(ctx, proposal.CommitteeID)
|
|
||||||
hasPermissions := false
|
|
||||||
for _, p := range committee.Permissions {
|
|
||||||
if p.Allows(proposal) {
|
|
||||||
hasPermissions = true
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
if !com.HasMember(proposer) {
|
||||||
|
return 0, sdk.ErrInternal("only member can propose proposals")
|
||||||
}
|
}
|
||||||
if !hasPermissions {
|
|
||||||
return sdk.ErrInternal("committee does not have permissions to enact proposal")
|
// Check proposal is valid
|
||||||
|
if err := proposal.ValidateBasic(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check group has permissions to enact proposal.
|
||||||
|
if !com.HasPermissionsFor(proposal) {
|
||||||
|
return 0, sdk.ErrInternal("committee does not have permissions to enact proposal")
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO validate proposal by running it with cached context like how gov does it
|
// TODO validate proposal by running it with cached context like how gov does it
|
||||||
|
// what if it's not valid now but will be in the future?
|
||||||
|
|
||||||
// TODO store the proposal, probably put it in a queue
|
// Get a new ID and store the proposal
|
||||||
|
return k.StoreNewProposal(ctx, proposal)
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (k Keeper) AddVote(ctx sdk.Context, msg types.MsgVote) sdk.Error {
|
func (k Keeper) AddVote(ctx sdk.Context, proposalID uint64, voter sdk.AccAddress) sdk.Error {
|
||||||
/* TODO
|
// Validate
|
||||||
- validate vote
|
proposal, found := k.GetProposal(ctx, proposalID)
|
||||||
- store vote
|
if !found {
|
||||||
*/
|
return sdk.ErrInternal("proposal not found")
|
||||||
|
}
|
||||||
|
com, found := k.GetCommittee(ctx, proposal.CommitteeID)
|
||||||
|
if !found {
|
||||||
|
return sdk.ErrInternal("committee disbanded")
|
||||||
|
}
|
||||||
|
if !com.HasMember(voter) {
|
||||||
|
return sdk.ErrInternal("not authorized to vote on proposal")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store vote, overwriting any prior vote
|
||||||
|
k.SetVote(ctx, types.Vote{ProposalID: proposalID, Voter: voter})
|
||||||
|
|
||||||
|
// TODO close vote if tally has been reached
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -96,6 +98,49 @@ func (k Keeper) DeleteCommittee(ctx sdk.Context, committeeID uint64) {
|
|||||||
store.Delete(types.GetKeyFromID(committeeID))
|
store.Delete(types.GetKeyFromID(committeeID))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetNextProposalID stores an ID to be used for the next created proposal
|
||||||
|
func (k Keeper) SetNextProposalID(ctx sdk.Context, id uint64) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
store.Set(types.NextProposalIDKey, types.GetKeyFromID(id))
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetNextProposalID reads the next available global ID from store
|
||||||
|
func (k Keeper) GetNextProposalID(ctx sdk.Context) (uint64, sdk.Error) {
|
||||||
|
store := ctx.KVStore(k.storeKey)
|
||||||
|
bz := store.Get(types.NextProposalIDKey)
|
||||||
|
if bz == nil {
|
||||||
|
return 0, sdk.ErrInternal("proposal ID not set at genesis")
|
||||||
|
}
|
||||||
|
return types.Uint64FromBytes(bz), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// IncrementNextProposalID increments the next proposal ID in the store by 1.
|
||||||
|
func (k Keeper) IncrementNextProposalID(ctx sdk.Context) sdk.Error {
|
||||||
|
id, err := k.GetNextProposalID(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
k.SetNextProposalID(ctx, id+1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// StoreNewProposal stores a proposal, adding a new ID
|
||||||
|
func (k Keeper) StoreNewProposal(ctx sdk.Context, proposal types.Proposal) (uint64, sdk.Error) {
|
||||||
|
newProposalID, err := k.GetNextProposalID(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
proposal.ID = newProposalID
|
||||||
|
|
||||||
|
k.SetProposal(ctx, proposal)
|
||||||
|
|
||||||
|
err = k.IncrementNextProposalID(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return newProposalID, nil
|
||||||
|
}
|
||||||
|
|
||||||
// GetProposal gets a proposal from the store.
|
// GetProposal gets a proposal from the store.
|
||||||
func (k Keeper) GetProposal(ctx sdk.Context, proposalID uint64) (types.Proposal, bool) {
|
func (k Keeper) GetProposal(ctx sdk.Context, proposalID uint64) (types.Proposal, bool) {
|
||||||
store := prefix.NewStore(ctx.KVStore(k.storeKey), types.ProposalKeyPrefix)
|
store := prefix.NewStore(ctx.KVStore(k.storeKey), types.ProposalKeyPrefix)
|
||||||
|
@ -24,15 +24,60 @@ type KeeperTestSuite struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (suite *KeeperTestSuite) SetupTest() {
|
func (suite *KeeperTestSuite) SetupTest() {
|
||||||
|
|
||||||
suite.app = app.NewTestApp()
|
suite.app = app.NewTestApp()
|
||||||
suite.keeper = suite.app.GetCommitteeKeeper()
|
suite.keeper = suite.app.GetCommitteeKeeper()
|
||||||
suite.ctx = suite.app.NewContext(true, abci.Header{})
|
suite.ctx = suite.app.NewContext(true, abci.Header{})
|
||||||
_, suite.addresses = app.GeneratePrivKeyAddressPairs(2)
|
_, suite.addresses = app.GeneratePrivKeyAddressPairs(2)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestSubmitProposal() {
|
||||||
|
testcases := []struct {
|
||||||
|
name string
|
||||||
|
proposal types.Proposal
|
||||||
|
proposer sdk.AccAddress
|
||||||
|
expectPass bool
|
||||||
|
}{
|
||||||
|
{name: "empty proposal", proposer: suite.addresses[0], expectPass: false},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testcases {
|
||||||
|
suite.Run(tc.name, func() {
|
||||||
|
_, err := suite.keeper.SubmitProposal(suite.ctx, tc.proposer, tc.proposal)
|
||||||
|
if tc.expectPass {
|
||||||
|
suite.NoError(err)
|
||||||
|
// TODO suite.keeper.GetProposal(suite.ctx, tc.proposal.ID)
|
||||||
|
} else {
|
||||||
|
suite.NotNil(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestAddVote() {
|
||||||
|
testcases := []struct {
|
||||||
|
name string
|
||||||
|
proposalID uint64
|
||||||
|
voter sdk.AccAddress
|
||||||
|
expectPass bool
|
||||||
|
}{
|
||||||
|
{name: "no proposal", proposalID: 9999999, voter: suite.addresses[0], expectPass: false},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testcases {
|
||||||
|
suite.Run(tc.name, func() {
|
||||||
|
err := suite.keeper.AddVote(suite.ctx, tc.proposalID, tc.voter)
|
||||||
|
if tc.expectPass {
|
||||||
|
suite.NoError(err)
|
||||||
|
// TODO GetVote
|
||||||
|
} else {
|
||||||
|
suite.NotNil(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (suite *KeeperTestSuite) TestGetSetDeleteCommittee() {
|
func (suite *KeeperTestSuite) TestGetSetDeleteCommittee() {
|
||||||
// test setup
|
// setup test
|
||||||
com := types.Committee{
|
com := types.Committee{
|
||||||
ID: 12,
|
ID: 12,
|
||||||
// TODO other fields
|
// TODO other fields
|
||||||
|
@ -32,7 +32,7 @@ var (
|
|||||||
VoteKeyPrefix = []byte{0x02} // prefix for keys that store votes
|
VoteKeyPrefix = []byte{0x02} // prefix for keys that store votes
|
||||||
//AuctionByTimeKeyPrefix = []byte{0x01} // prefix for keys that are part of the auctionsByTime index
|
//AuctionByTimeKeyPrefix = []byte{0x01} // prefix for keys that are part of the auctionsByTime index
|
||||||
|
|
||||||
//NextAuctionIDKey = []byte{0x02} // key for the next auction id
|
NextProposalIDKey = []byte{0x03} // key for the next proposal id
|
||||||
)
|
)
|
||||||
|
|
||||||
// GetKeyFromID returns the bytes to use as a key for a uint64 id
|
// GetKeyFromID returns the bytes to use as a key for a uint64 id
|
||||||
@ -57,6 +57,6 @@ func uint64ToBytes(id uint64) []byte {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Uint64FromBytes converts some fixed length bytes back into a uint64.
|
// Uint64FromBytes converts some fixed length bytes back into a uint64.
|
||||||
func uint64FromBytes(bz []byte) uint64 {
|
func Uint64FromBytes(bz []byte) uint64 {
|
||||||
return binary.BigEndian.Uint64(bz)
|
return binary.BigEndian.Uint64(bz)
|
||||||
}
|
}
|
||||||
|
@ -12,6 +12,25 @@ type Committee struct {
|
|||||||
Permissions []Permission
|
Permissions []Permission
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c Committee) HasMember(addr sdk.AccAddress) bool {
|
||||||
|
for _, m := range c.Members {
|
||||||
|
if m.Equals(addr) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// As long as one permission allows the proposal then it goes through. Its the OR of all permissions.
|
||||||
|
func (c Committee) HasPermissionsFor(proposal gov.Content) bool {
|
||||||
|
for _, p := range c.Permissions {
|
||||||
|
if p.Allows(proposal) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
// Permission is anything with a method that validates whether a proposal is allowed by it or not.
|
// Permission is anything with a method that validates whether a proposal is allowed by it or not.
|
||||||
type Permission interface {
|
type Permission interface {
|
||||||
Allows(gov.Content) bool
|
Allows(gov.Content) bool
|
||||||
@ -20,6 +39,8 @@ type Permission interface {
|
|||||||
// GOV STUFF --------------------------
|
// GOV STUFF --------------------------
|
||||||
// Should be much the same as in gov module, except Proposals are linked to a committee ID.
|
// Should be much the same as in gov module, except Proposals are linked to a committee ID.
|
||||||
|
|
||||||
|
var _ gov.Content = Proposal{}
|
||||||
|
|
||||||
type Proposal struct {
|
type Proposal struct {
|
||||||
gov.Content
|
gov.Content
|
||||||
ID uint64
|
ID uint64
|
||||||
@ -31,7 +52,7 @@ type Proposal struct {
|
|||||||
type Vote struct {
|
type Vote struct {
|
||||||
ProposalID uint64
|
ProposalID uint64
|
||||||
Voter sdk.AccAddress
|
Voter sdk.AccAddress
|
||||||
Option byte
|
// Option byte // TODO for now don't need more than just a yes as options
|
||||||
}
|
}
|
||||||
|
|
||||||
// Genesis -------------------
|
// Genesis -------------------
|
||||||
|
Loading…
Reference in New Issue
Block a user