Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: dyncomm module #355

Merged
merged 32 commits into from
Nov 4, 2023
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
1843d12
feat: dyncomm first boilerplate
fragwuerdig Sep 22, 2023
864a567
fix: delete duplicate query.proto
fragwuerdig Sep 22, 2023
287ce7a
fix: no module account for dyncomm
fragwuerdig Sep 22, 2023
3eac75c
fix: inject staking instead of account keeper
fragwuerdig Sep 22, 2023
c5786be
add: dyncomm.go routines and param getters
fragwuerdig Sep 22, 2023
927b49f
fix: miscalculation of dyn. commission rate
fragwuerdig Sep 23, 2023
191ea40
fix: return dyncomm as fraction
fragwuerdig Sep 23, 2023
3f94b8d
fix: paramspace get accepts pointer
fragwuerdig Sep 23, 2023
25bd177
finalize core logic of the module
fragwuerdig Sep 23, 2023
a6a2881
add: per-validator min commission store
fragwuerdig Sep 23, 2023
5178fea
add: ante handler checking MsgEditValidator
fragwuerdig Sep 24, 2023
a670236
add: create test mock
fragwuerdig Sep 25, 2023
704cd6c
fix: test utils mockup
fragwuerdig Sep 25, 2023
e520532
add: TestCalculateVotingPower()
fragwuerdig Sep 25, 2023
b35378d
add: TestCalculateDynCommission
fragwuerdig Sep 25, 2023
552a6c3
add cli for dyncomm (#359)
alchemist-ti Sep 26, 2023
201f0a0
fix: update min rates more often
fragwuerdig Sep 26, 2023
940f53f
add: add straths suggestions
fragwuerdig Sep 26, 2023
43c6797
Fix regression (#365)
StrathCole Oct 4, 2023
17541c7
fix: move dyncomm state changes to post handler
fragwuerdig Oct 6, 2023
d868d4b
fix: improve/fix parameter validation
fragwuerdig Oct 16, 2023
3124e54
fix: latest compiler messages
fragwuerdig Oct 16, 2023
faa765a
chore: adjust upgrade test to requirements of SDK v0.46
fragwuerdig Oct 16, 2023
fe3fb0e
add: upgrade handling
fragwuerdig Oct 18, 2023
5f18601
add: test to ensure ante handler
fragwuerdig Oct 21, 2023
51d5557
fix: workaround for staking simulation interfering with dyncomm
fragwuerdig Oct 21, 2023
57dca07
style: make lint-strict
fragwuerdig Oct 21, 2023
3529f01
fix: more linting
fragwuerdig Oct 21, 2023
c8b5c5e
prepare testnet release with 10mins epoch
fragwuerdig Oct 21, 2023
673cf03
style: another lint
fragwuerdig Oct 21, 2023
2c64e20
fix upgrade test
nghuyenthevinh2000 Nov 1, 2023
24c61c4
revert: target rate shall only be updated through create and edit val…
fragwuerdig Nov 1, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions app/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -211,6 +211,8 @@ func NewTerraApp(
GovKeeper: app.GovKeeper,
WasmConfig: &wasmConfig,
TXCounterStoreKey: app.GetKey(wasm.StoreKey),
DyncommKeeper: app.DyncommKeeper,
StakingKeeper: app.StakingKeeper,
},
)
if err != nil {
Expand Down
12 changes: 12 additions & 0 deletions app/keepers/keepers.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,8 @@ import (
customwasmkeeper "github.com/classic-terra/core/v2/custom/wasm/keeper"
terrawasm "github.com/classic-terra/core/v2/wasmbinding"

dyncommkeeper "github.com/classic-terra/core/v2/x/dyncomm/keeper"
dyncommtypes "github.com/classic-terra/core/v2/x/dyncomm/types"
marketkeeper "github.com/classic-terra/core/v2/x/market/keeper"
markettypes "github.com/classic-terra/core/v2/x/market/types"
oraclekeeper "github.com/classic-terra/core/v2/x/oracle/keeper"
Expand Down Expand Up @@ -92,6 +94,7 @@ type AppKeepers struct {
MarketKeeper marketkeeper.Keeper
TreasuryKeeper treasurykeeper.Keeper
WasmKeeper wasmkeeper.Keeper
DyncommKeeper dyncommkeeper.Keeper

// make scoped keepers public for test purposes
ScopedIBCKeeper capabilitykeeper.ScopedKeeper
Expand Down Expand Up @@ -137,6 +140,7 @@ func NewAppKeepers(
markettypes.StoreKey,
treasurytypes.StoreKey,
wasmtypes.StoreKey,
dyncommtypes.StoreKey,
)
tkeys := sdk.NewTransientStoreKeys(paramstypes.TStoreKey)
memKeys := sdk.NewMemoryStoreKeys(capabilitytypes.MemStoreKey)
Expand Down Expand Up @@ -407,6 +411,13 @@ func NewAppKeepers(
govConfig,
)

appKeepers.DyncommKeeper = dyncommkeeper.NewKeeper(
appCodec,
appKeepers.keys[dyncommtypes.StoreKey],
appKeepers.GetSubspace(dyncommtypes.ModuleName),
appKeepers.StakingKeeper,
)

appKeepers.ScopedIBCKeeper = scopedIBCKeeper
appKeepers.ScopedICAHostKeeper = scopedICAHostKeeper
appKeepers.ScopedICAControllerKeeper = scopedICAControllerKeeper
Expand Down Expand Up @@ -442,6 +453,7 @@ func initParamsKeeper(
paramsKeeper.Subspace(oracletypes.ModuleName)
paramsKeeper.Subspace(treasurytypes.ModuleName)
paramsKeeper.Subspace(wasmtypes.ModuleName)
paramsKeeper.Subspace(dyncommtypes.ModuleName)

return paramsKeeper
}
Expand Down
8 changes: 8 additions & 0 deletions app/modules.go
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,8 @@ import (
"github.com/CosmWasm/wasmd/x/wasm"
wasmclient "github.com/CosmWasm/wasmd/x/wasm/client"
wasmtypes "github.com/CosmWasm/wasmd/x/wasm/types"
"github.com/classic-terra/core/v2/x/dyncomm"
dyncommtypes "github.com/classic-terra/core/v2/x/dyncomm/types"
"github.com/classic-terra/core/v2/x/market"
markettypes "github.com/classic-terra/core/v2/x/market/types"
"github.com/classic-terra/core/v2/x/oracle"
Expand Down Expand Up @@ -121,6 +123,7 @@ var (
treasury.AppModuleBasic{},
customwasm.AppModuleBasic{},
ibcfee.AppModuleBasic{},
dyncomm.AppModuleBasic{},
)

// module account permissions
Expand Down Expand Up @@ -182,6 +185,7 @@ func appModules(
oracle.NewAppModule(appCodec, app.OracleKeeper, app.AccountKeeper, app.BankKeeper),
treasury.NewAppModule(appCodec, app.TreasuryKeeper),
wasm.NewAppModule(appCodec, &app.WasmKeeper, app.StakingKeeper, app.AccountKeeper, app.BankKeeper),
dyncomm.NewAppModule(appCodec, app.DyncommKeeper, app.StakingKeeper),
}
}

Expand Down Expand Up @@ -213,6 +217,7 @@ func simulationModules(
market.NewAppModule(appCodec, app.MarketKeeper, app.AccountKeeper, app.BankKeeper, app.OracleKeeper),
treasury.NewAppModule(appCodec, app.TreasuryKeeper),
wasm.NewAppModule(appCodec, &app.WasmKeeper, app.StakingKeeper, app.AccountKeeper, app.BankKeeper),
dyncomm.NewAppModule(appCodec, app.DyncommKeeper, app.StakingKeeper),
}
}

Expand Down Expand Up @@ -243,6 +248,7 @@ func orderBeginBlockers() []string {
treasurytypes.ModuleName,
markettypes.ModuleName,
wasmtypes.ModuleName,
dyncommtypes.ModuleName,
}
}

Expand Down Expand Up @@ -273,6 +279,7 @@ func orderEndBlockers() []string {
treasurytypes.ModuleName,
markettypes.ModuleName,
wasmtypes.ModuleName,
dyncommtypes.ModuleName,
}
}

Expand Down Expand Up @@ -303,5 +310,6 @@ func orderInitGenesis() []string {
oracletypes.ModuleName,
treasurytypes.ModuleName,
wasmtypes.ModuleName,
dyncommtypes.ModuleName,
}
}
6 changes: 6 additions & 0 deletions custom/auth/ante/ante.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
package ante

import (
dyncommante "github.com/classic-terra/core/v2/x/dyncomm/ante"
dyncommkeeper "github.com/classic-terra/core/v2/x/dyncomm/keeper"
stakingkeeper "github.com/cosmos/cosmos-sdk/x/staking/keeper"
ibcante "github.com/cosmos/ibc-go/v6/modules/core/ante"
ibckeeper "github.com/cosmos/ibc-go/v6/modules/core/keeper"

Expand Down Expand Up @@ -31,6 +34,8 @@ type HandlerOptions struct {
GovKeeper govkeeper.Keeper
WasmConfig *wasmtypes.WasmConfig
TXCounterStoreKey storetypes.StoreKey
DyncommKeeper dyncommkeeper.Keeper
StakingKeeper stakingkeeper.Keeper
}

// NewAnteHandler returns an AnteHandler that checks and increments sequence
Expand Down Expand Up @@ -85,5 +90,6 @@ func NewAnteHandler(options HandlerOptions) (sdk.AnteHandler, error) {
ante.NewSigVerificationDecorator(options.AccountKeeper, options.SignModeHandler),
ante.NewIncrementSequenceDecorator(options.AccountKeeper),
ibcante.NewRedundantRelayDecorator(&options.IBCKeeper),
dyncommante.NewDyncommDecorator(options.DyncommKeeper, options.StakingKeeper),
), nil
}
36 changes: 36 additions & 0 deletions proto/terra/dyncomm/v1beta1/dyncomm.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
syntax = "proto3";
package terra.dyncomm.v1beta1;

import "gogoproto/gogo.proto";

option go_package = "github.com/classic-terra/core/v2/x/dyncomm/types";

// Params defines the parameters for the dyncomm module.
message Params {
option (gogoproto.equal) = true;
option (gogoproto.goproto_stringer) = false;

string max_zero = 1 [
(gogoproto.moretags) = "yaml:\"max_zero\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];

string slope_base = 2 [
(gogoproto.moretags) = "yaml:\"slope_base\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];

string slope_vp_impact = 3 [
(gogoproto.moretags) = "yaml:\"slope_vp_impact\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];

string cap = 4 [
(gogoproto.moretags) = "yaml:\"cap\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
}
21 changes: 21 additions & 0 deletions proto/terra/dyncomm/v1beta1/genesis.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
syntax = "proto3";
package terra.dyncomm.v1beta1;

import "gogoproto/gogo.proto";
import "terra/dyncomm/v1beta1/dyncomm.proto";

option go_package = "github.com/classic-terra/core/v2/x/dyncomm/types";

// GenesisState defines the dyncomm module's genesis state.
message GenesisState {
// params defines all the paramaters of the module.
Params params = 1 [(gogoproto.nullable) = false];
repeated MinCommissionRate min_commission_rates = 2 [(gogoproto.nullable) = false];
}

// MinDynCommission defines a validator - min commission rate
// pair to be enforced by the blockchain
message MinCommissionRate {
string validator_address = 1;
string min_commission_rate = 2 [(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec"];
}
25 changes: 25 additions & 0 deletions proto/terra/dyncomm/v1beta1/query.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
syntax = "proto3";
package terra.dyncomm.v1beta1;

import "gogoproto/gogo.proto";
import "google/api/annotations.proto";
import "terra/dyncomm/v1beta1/dyncomm.proto";

option go_package = "github.com/classic-terra/core/v2/x/dyncomm/types";

// Query defines the gRPC querier service.
service Query {
// Params queries all parameters.
rpc Params(QueryParamsRequest) returns (QueryParamsResponse) {
option (google.api.http).get = "/terra/dyncomm/v1beta1/params";
}
}

// QueryParamsRequest is the request type for the Query/Params RPC method.
message QueryParamsRequest {}

// QueryParamsResponse is the response type for the Query/Params RPC method.
message QueryParamsResponse {
// params defines the parameters of the module.
Params params = 1 [(gogoproto.nullable) = false];
}
28 changes: 28 additions & 0 deletions x/dyncomm/abci.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
package dyncomm

import (
"time"

"github.com/cosmos/cosmos-sdk/telemetry"
sdk "github.com/cosmos/cosmos-sdk/types"

"github.com/classic-terra/core/v2/x/dyncomm/keeper"
"github.com/classic-terra/core/v2/x/dyncomm/types"

core "github.com/classic-terra/core/v2/types"
)

// EndBlocker is called at the end of every block
func EndBlocker(ctx sdk.Context, k keeper.Keeper) {
defer telemetry.ModuleMeasureSince(types.ModuleName, time.Now(), telemetry.MetricKeyEndBlocker)

// Check epoch last block
// TODO: Change this to the appropriate period
if !core.IsPeriodLastBlock(ctx, core.BlocksPerMinute) {
return
}

ctx.Logger().Info("End Epoch - Calculation of Dyncomm is due")
k.UpdateAllBondedValidatorRates(ctx)

}
84 changes: 84 additions & 0 deletions x/dyncomm/ante/ante.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
package ante

import (
"fmt"

dyncommkeeper "github.com/classic-terra/core/v2/x/dyncomm/keeper"
sdk "github.com/cosmos/cosmos-sdk/types"
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
stakingkeeper "github.com/cosmos/cosmos-sdk/x/staking/keeper"
stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types"
)

// DyncommDecorator checks for EditValidator and rejects
// edits that do not conform with dyncomm
type DyncommDecorator struct {
dyncommKeeper dyncommkeeper.Keeper
stakingKeeper stakingkeeper.Keeper
}

func NewDyncommDecorator(dk dyncommkeeper.Keeper, sk stakingkeeper.Keeper) DyncommDecorator {
return DyncommDecorator{
dyncommKeeper: dk,
stakingKeeper: sk,
}
}

// IsMsgSubmitProposal checks whether the input msg is a MsgSubmitProposal
func IsMsgEditValidator(msg sdk.Msg) bool {
_, ok := msg.(*stakingtypes.MsgEditValidator)
return ok
}

func (dd DyncommDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simulate bool, next sdk.AnteHandler) (sdk.Context, error) {
if simulate {
return next(ctx, tx, simulate)
}

msgs := tx.GetMsgs()
err := dd.FilterMsgsAndCheckEditValidator(ctx, msgs...)
fragwuerdig marked this conversation as resolved.
Show resolved Hide resolved

if err != nil {
return ctx, err
}

return next(ctx, tx, simulate)

}

func (dd DyncommDecorator) FilterMsgsAndCheckEditValidator(ctx sdk.Context, msgs ...sdk.Msg) (err error) {

for _, msg := range msgs {
if !IsMsgEditValidator(msg) {
continue
}

err := dd.CheckEditValidator(ctx, msg)
if err != nil {
return sdkerrors.Wrapf(sdkerrors.ErrUnauthorized, err.Error())
}
}
return nil

}

func (dd DyncommDecorator) CheckEditValidator(ctx sdk.Context, msg sdk.Msg) (err error) {

msgEditValidator := msg.(*stakingtypes.MsgEditValidator)

// no update of CommissionRate provided
if msgEditValidator.CommissionRate == nil {
return nil
}

operator := msgEditValidator.ValidatorAddress
newIntendedRate := msgEditValidator.CommissionRate
dynMinRate := dd.dyncommKeeper.GetDynCommissionRate(ctx, operator)

if newIntendedRate.LT(dynMinRate) {
return fmt.Errorf("commission for %s must be at least %f", operator, dynMinRate.MustFloat64())
}

return nil

}
35 changes: 35 additions & 0 deletions x/dyncomm/genesis.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package dyncomm

import (
sdk "github.com/cosmos/cosmos-sdk/types"

"github.com/classic-terra/core/v2/x/dyncomm/keeper"
"github.com/classic-terra/core/v2/x/dyncomm/types"
)

// InitGenesis initializes default parameters
// and the keeper's address to pubkey map
func InitGenesis(ctx sdk.Context, keeper keeper.Keeper, data *types.GenesisState) {
keeper.SetParams(ctx, data.Params)

for _, commRate := range data.MinCommissionRates {
keeper.SetDynCommissionRate(ctx, commRate.ValidatorAddress, *commRate.MinCommissionRate)
}

}

// ExportGenesis writes the current store values
// to a genesis file, which can be imported again
// with InitGenesis
func ExportGenesis(ctx sdk.Context, keeper keeper.Keeper) (data *types.GenesisState) {
params := keeper.GetParams(ctx)
var rates []types.MinCommissionRate

rates = append(rates)
keeper.IterateDynCommissionRates(ctx, func(rate types.MinCommissionRate) (stop bool) {
rates = append(rates, rate)
return false
})

return types.NewGenesisState(params, rates)
}
Loading