Skip to content

Commit

Permalink
Merge branch 'v0.45.16-ics-lsm' into sam/stride-audit-fixes
Browse files Browse the repository at this point in the history
  • Loading branch information
sampocs committed Jun 27, 2023
2 parents cb150a5 + dc5ab59 commit 6ebf1a5
Show file tree
Hide file tree
Showing 18 changed files with 1,679 additions and 1,182 deletions.
21 changes: 20 additions & 1 deletion docs/core/proto-docs.md
Original file line number Diff line number Diff line change
Expand Up @@ -493,6 +493,7 @@
- [cosmos/staking/v1beta1/genesis.proto](#cosmos/staking/v1beta1/genesis.proto)
- [GenesisState](#cosmos.staking.v1beta1.GenesisState)
- [LastValidatorPower](#cosmos.staking.v1beta1.LastValidatorPower)
- [TokenizeShareLock](#cosmos.staking.v1beta1.TokenizeShareLock)

- [cosmos/staking/v1beta1/query.proto](#cosmos/staking/v1beta1/query.proto)
- [QueryAllTokenizeShareRecordsRequest](#cosmos.staking.v1beta1.QueryAllTokenizeShareRecordsRequest)
Expand Down Expand Up @@ -6766,7 +6767,6 @@ Params defines the parameters for the staking module.
| `max_entries` | [uint32](#uint32) | | max_entries is the max entries for either unbonding delegation or redelegation (per pair/trio). |
| `historical_entries` | [uint32](#uint32) | | historical_entries is the number of historical entries to persist. |
| `bond_denom` | [string](#string) | | bond_denom defines the bondable coin denomination. |
| `min_commission_rate` | [string](#string) | | min_commission_rate is the chain-wide minimum commission rate that a validator can charge their delegators |
| `validator_bond_factor` | [string](#string) | | validator_bond_factor is required as a safety check for tokenizing shares and delegations from liquid staking providers |
| `global_liquid_staking_cap` | [string](#string) | | global_liquid_staking_cap represents a cap on the portion of stake that comes from liquid staking providers |
| `validator_liquid_staking_cap` | [string](#string) | | validator_liquid_staking_cap represents a cap on the portion of stake that comes from liquid staking providers for a specific validator |
Expand Down Expand Up @@ -7084,6 +7084,8 @@ GenesisState defines the staking module's genesis state.
| `exported` | [bool](#bool) | | |
| `tokenize_share_records` | [TokenizeShareRecord](#cosmos.staking.v1beta1.TokenizeShareRecord) | repeated | store tokenize share records to provide reward to record owners |
| `last_tokenize_share_record_id` | [uint64](#uint64) | | last tokenize share record id, used for next share record id calculation |
| `total_liquid_staked_tokens` | [bytes](#bytes) | | total number of liquid staked tokens at genesis |
| `tokenize_share_locks` | [TokenizeShareLock](#cosmos.staking.v1beta1.TokenizeShareLock) | repeated | tokenize shares locks at genesis |



Expand All @@ -7105,6 +7107,23 @@ LastValidatorPower required for validator set update logic.




<a name="cosmos.staking.v1beta1.TokenizeShareLock"></a>

### TokenizeShareLock
TokenizeSharesLock required for specifying account locks at genesis


| Field | Type | Label | Description |
| ----- | ---- | ----- | ----------- |
| `address` | [string](#string) | | Address of the account that is locked |
| `status` | [string](#string) | | Status of the lock (LOCKED or LOCK_EXPIRING) |
| `completion_time` | [google.protobuf.Timestamp](#google.protobuf.Timestamp) | | Completion time if the lock is expiring |





<!-- end messages -->

<!-- end enums -->
Expand Down
22 changes: 22 additions & 0 deletions proto/cosmos/staking/v1beta1/genesis.proto
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ option go_package = "github.com/cosmos/cosmos-sdk/x/staking/types";

import "gogoproto/gogo.proto";
import "cosmos/staking/v1beta1/staking.proto";
import "google/protobuf/timestamp.proto";

// GenesisState defines the staking module's genesis state.
message GenesisState {
Expand Down Expand Up @@ -44,6 +45,27 @@ message GenesisState {

// last tokenize share record id, used for next share record id calculation
uint64 last_tokenize_share_record_id = 10;

// total number of liquid staked tokens at genesis
bytes total_liquid_staked_tokens = 11 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Int",
(gogoproto.moretags) = "yaml:\"total_liquid_staked_tokens\"",
(gogoproto.nullable) = false
];

// tokenize shares locks at genesis
repeated TokenizeShareLock tokenize_share_locks = 12 [(gogoproto.nullable) = false];
}

// TokenizeSharesLock required for specifying account locks at genesis
message TokenizeShareLock {
// Address of the account that is locked
string address = 1;
// Status of the lock (LOCKED or LOCK_EXPIRING)
string status = 2;
// Completion time if the lock is expiring
google.protobuf.Timestamp completion_time = 3
[(gogoproto.nullable) = false, (gogoproto.stdtime) = true, (gogoproto.moretags) = "yaml:\"completion_time\""];
}

// LastValidatorPower required for validator set update logic.
Expand Down
12 changes: 3 additions & 9 deletions proto/cosmos/staking/v1beta1/staking.proto
Original file line number Diff line number Diff line change
Expand Up @@ -308,29 +308,23 @@ message Params {
uint32 historical_entries = 4 [(gogoproto.moretags) = "yaml:\"historical_entries\""];
// bond_denom defines the bondable coin denomination.
string bond_denom = 5 [(gogoproto.moretags) = "yaml:\"bond_denom\""];
// min_commission_rate is the chain-wide minimum commission rate that a validator can charge their delegators
string min_commission_rate = 6 [
(gogoproto.moretags) = "yaml:\"min_commission_rate\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// validator_bond_factor is required as a safety check for tokenizing shares and
// delegations from liquid staking providers
string validator_bond_factor = 7 [
string validator_bond_factor = 6 [
(gogoproto.moretags) = "yaml:\"validator_bond_factor\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// global_liquid_staking_cap represents a cap on the portion of stake that
// comes from liquid staking providers
string global_liquid_staking_cap = 8 [
string global_liquid_staking_cap = 7 [
(gogoproto.moretags) = "yaml:\"global_liquid_staking_cap\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// validator_liquid_staking_cap represents a cap on the portion of stake that
// comes from liquid staking providers for a specific validator
string validator_liquid_staking_cap = 9 [
string validator_liquid_staking_cap = 8 [
(gogoproto.moretags) = "yaml:\"validator_liquid_staking_cap\"",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
Expand Down
3 changes: 1 addition & 2 deletions x/staking/client/testutil/suite.go
Original file line number Diff line number Diff line change
Expand Up @@ -895,15 +895,14 @@ global_liquid_staking_cap: "1.000000000000000000"
historical_entries: 10000
max_entries: 7
max_validators: 100
min_commission_rate: "0.000000000000000000"
unbonding_time: 1814400s
validator_bond_factor: "-1.000000000000000000"
validator_liquid_staking_cap: "1.000000000000000000"`,
},
{
"with json output",
[]string{fmt.Sprintf("--%s=json", tmcli.OutputFlag)},
`{"unbonding_time":"1814400s","max_validators":100,"max_entries":7,"historical_entries":10000,"bond_denom":"stake","min_commission_rate":"0.000000000000000000","validator_bond_factor":"-1.000000000000000000","global_liquid_staking_cap":"1.000000000000000000","validator_liquid_staking_cap":"1.000000000000000000"}`,
`{"unbonding_time":"1814400s","max_validators":100,"max_entries":7,"historical_entries":10000,"bond_denom":"stake","validator_bond_factor":"-1.000000000000000000","global_liquid_staking_cap":"1.000000000000000000","validator_liquid_staking_cap":"1.000000000000000000"}`,
},
}
for _, tc := range testCases {
Expand Down
62 changes: 54 additions & 8 deletions x/staking/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -150,6 +150,48 @@ func InitGenesis(
}
}

// Set the total liquid staked tokens
keeper.SetTotalLiquidStakedTokens(ctx, data.TotalLiquidStakedTokens)

// Set each tokenize share record, as well as the last tokenize share record ID
latestId := uint64(0)
for _, tokenizeShareRecord := range data.TokenizeShareRecords {
if err := keeper.AddTokenizeShareRecord(ctx, tokenizeShareRecord); err != nil {
panic(err)
}
if tokenizeShareRecord.Id > latestId {
latestId = tokenizeShareRecord.Id
}
}
if data.LastTokenizeShareRecordId < latestId {
panic("Tokenize share record specified with ID greater than the latest ID")
}
keeper.SetLastTokenizeShareRecordID(ctx, data.LastTokenizeShareRecordId)

// Set the tokenize shares locks for accounts that have disabled tokenizing shares
// The lock can either be in status LOCKED or LOCK_EXPIRING
// If it is in status LOCK_EXPIRING, a the unlocking must also be queued
for _, tokenizeShareLock := range data.TokenizeShareLocks {
address := sdk.MustAccAddressFromBech32(tokenizeShareLock.Address)

switch tokenizeShareLock.Status {
case types.TOKENIZE_SHARE_LOCK_STATUS_LOCKED.String():
keeper.AddTokenizeSharesLock(ctx, address)

case types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING.String():
completionTime := tokenizeShareLock.CompletionTime

authorizations := keeper.GetPendingTokenizeShareAuthorizations(ctx, completionTime)
authorizations.Addresses = append(authorizations.Addresses, address.String())

keeper.SetPendingTokenizeShareAuthorizations(ctx, completionTime, authorizations)
keeper.SetTokenizeSharesUnlockTime(ctx, address, completionTime)

default:
panic(fmt.Sprintf("Unsupported tokenize share lock status %s", tokenizeShareLock.Status))
}
}

return res
}

Expand Down Expand Up @@ -179,14 +221,18 @@ func ExportGenesis(ctx sdk.Context, keeper keeper.Keeper) *types.GenesisState {
})

return &types.GenesisState{
Params: keeper.GetParams(ctx),
LastTotalPower: keeper.GetLastTotalPower(ctx),
LastValidatorPowers: lastValidatorPowers,
Validators: keeper.GetAllValidators(ctx),
Delegations: keeper.GetAllDelegations(ctx),
UnbondingDelegations: unbondingDelegations,
Redelegations: redelegations,
Exported: true,
Params: keeper.GetParams(ctx),
LastTotalPower: keeper.GetLastTotalPower(ctx),
LastValidatorPowers: lastValidatorPowers,
Validators: keeper.GetAllValidators(ctx),
Delegations: keeper.GetAllDelegations(ctx),
UnbondingDelegations: unbondingDelegations,
Redelegations: redelegations,
Exported: true,
TokenizeShareRecords: keeper.GetAllTokenizeShareRecords(ctx),
LastTokenizeShareRecordId: keeper.GetLastTokenizeShareRecordID(ctx),
TotalLiquidStakedTokens: keeper.GetTotalLiquidStakedTokens(ctx),
TokenizeShareLocks: keeper.GetAllTokenizeSharesLocks(ctx),
}
}

Expand Down
47 changes: 47 additions & 0 deletions x/staking/genesis_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import (
"fmt"
"log"
"testing"
"time"

"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
Expand Down Expand Up @@ -242,3 +243,49 @@ func TestValidateGenesis(t *testing.T) {
})
}
}

func TestInitExportLiquidStakingGenesis(t *testing.T) {
app := simapp.Setup(false)
ctx := app.NewContext(false, tmproto.Header{})

addresses := simapp.AddTestAddrs(app, ctx, 2, sdk.OneInt())
address1, address2 := addresses[0], addresses[1]

// Mock out a genesis state
inGenesisState := types.GenesisState{
Params: types.DefaultParams(),
TokenizeShareRecords: []types.TokenizeShareRecord{
{Id: 1, Owner: address1.String(), ModuleAccount: "module1", Validator: "val1"},
{Id: 2, Owner: address2.String(), ModuleAccount: "module2", Validator: "val2"},
},
LastTokenizeShareRecordId: 2,
TotalLiquidStakedTokens: sdk.NewInt(1_000_000),
TokenizeShareLocks: []types.TokenizeShareLock{
{
Address: address1.String(),
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCKED.String(),
},
{
Address: address2.String(),
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING.String(),
CompletionTime: time.Date(2023, 1, 1, 1, 0, 0, 0, time.UTC),
},
},
}

// Call init and then export genesis - confirming the same state is returned
staking.InitGenesis(ctx, app.StakingKeeper, app.AccountKeeper, app.BankKeeper, &inGenesisState)
outGenesisState := *staking.ExportGenesis(ctx, app.StakingKeeper)

require.ElementsMatch(t, inGenesisState.TokenizeShareRecords, outGenesisState.TokenizeShareRecords,
"tokenize share records")

require.Equal(t, inGenesisState.LastTokenizeShareRecordId, outGenesisState.LastTokenizeShareRecordId,
"last tokenize share record ID")

require.Equal(t, inGenesisState.TotalLiquidStakedTokens.Int64(), outGenesisState.TotalLiquidStakedTokens.Int64(),
"total liquid staked")

require.ElementsMatch(t, inGenesisState.TokenizeShareLocks, outGenesisState.TokenizeShareLocks,
"tokenize share locks")
}
36 changes: 35 additions & 1 deletion x/staking/keeper/liquid_stake.go
Original file line number Diff line number Diff line change
Expand Up @@ -210,6 +210,40 @@ func (k Keeper) GetTokenizeSharesLock(ctx sdk.Context, address sdk.AccAddress) (
return types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING, unlockTime
}

// Returns all tokenize share locks
func (k Keeper) GetAllTokenizeSharesLocks(ctx sdk.Context) (tokenizeShareLocks []types.TokenizeShareLock) {
store := ctx.KVStore(k.storeKey)

iterator := sdk.KVStorePrefixIterator(store, types.TokenizeSharesLockPrefix)
defer iterator.Close()

for ; iterator.Valid(); iterator.Next() {
addressBz := iterator.Key()[2:] // remove prefix bytes and address length
unlockTime, err := sdk.ParseTimeBytes(iterator.Value())
if err != nil {
panic(err)
}

var status types.TokenizeShareLockStatus
if unlockTime.IsZero() {
status = types.TOKENIZE_SHARE_LOCK_STATUS_LOCKED
} else {
status = types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING
}

bechPrefix := sdk.GetConfig().GetBech32AccountAddrPrefix()
lock := types.TokenizeShareLock{
Address: sdk.MustBech32ifyAddressBytes(bechPrefix, addressBz),
Status: status.String(),
CompletionTime: unlockTime,
}

tokenizeShareLocks = append(tokenizeShareLocks, lock)
}

return tokenizeShareLocks
}

// Stores a list of addresses pending tokenize share unlocking at the same time
func (k Keeper) SetPendingTokenizeShareAuthorizations(ctx sdk.Context, completionTime time.Time, authorizations types.PendingTokenizeShareAuthorizations) {
store := ctx.KVStore(k.storeKey)
Expand Down Expand Up @@ -273,7 +307,7 @@ func (k Keeper) RemoveExpiredTokenizeShareLocks(ctx sdk.Context, blockTime time.

// iterators all time slices from time 0 until the current block time
prefixEnd := sdk.InclusiveEndBytes(types.GetTokenizeShareAuthorizationTimeKey(blockTime))
iterator := store.Iterator(types.TokenizeSharesUnlockQueueKey, prefixEnd)
iterator := store.Iterator(types.TokenizeSharesUnlockQueuePrefix, prefixEnd)
defer iterator.Close()

unlockedAddresses = []string{}
Expand Down
45 changes: 45 additions & 0 deletions x/staking/keeper/liquid_stake_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -773,6 +773,51 @@ func TestTokenizeSharesLock(t *testing.T) {
require.Equal(t, unlocked, status.String(), "addressB unlocked at end")
}

// Tests GetAllTokenizeSharesLocks
func TestGetAllTokenizeSharesLocks(t *testing.T) {
_, app, ctx := createTestInput()

addresses := simapp.AddTestAddrs(app, ctx, 4, sdk.NewInt(1))

// Set 2 locked accounts, and two accounts with a lock expiring
app.StakingKeeper.AddTokenizeSharesLock(ctx, addresses[0])
app.StakingKeeper.AddTokenizeSharesLock(ctx, addresses[1])

unlockTime1 := time.Date(2023, 1, 1, 1, 0, 0, 0, time.UTC)
unlockTime2 := time.Date(2023, 1, 2, 1, 0, 0, 0, time.UTC)
app.StakingKeeper.SetTokenizeSharesUnlockTime(ctx, addresses[2], unlockTime1)
app.StakingKeeper.SetTokenizeSharesUnlockTime(ctx, addresses[3], unlockTime2)

// Defined expected locks after GetAll
expectedLocks := map[string]types.TokenizeShareLock{
addresses[0].String(): {
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCKED.String(),
},
addresses[1].String(): {
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCKED.String(),
},
addresses[2].String(): {
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING.String(),
CompletionTime: unlockTime1,
},
addresses[3].String(): {
Status: types.TOKENIZE_SHARE_LOCK_STATUS_LOCK_EXPIRING.String(),
CompletionTime: unlockTime2,
},
}

// Check output from GetAll
actualLocks := app.StakingKeeper.GetAllTokenizeSharesLocks(ctx)
require.Len(t, actualLocks, len(expectedLocks), "number of locks")

for i, actual := range actualLocks {
expected, ok := expectedLocks[actual.Address]
require.True(t, ok, "address %s not expected", actual.Address)
require.Equal(t, expected.Status, actual.Status, "tokenize share lock #%d status", i)
require.Equal(t, expected.CompletionTime, actual.CompletionTime, "tokenize share lock #%d completion time", i)
}
}

// Test Get/SetPendingTokenizeShareAuthorizations
func TestPendingTokenizeShareAuthorizations(t *testing.T) {
_, app, ctx := createTestInput()
Expand Down
5 changes: 0 additions & 5 deletions x/staking/keeper/msg_server.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ package keeper

import (
"context"
"errors"
"fmt"
"strconv"
"time"
Expand Down Expand Up @@ -41,10 +40,6 @@ func (k msgServer) CreateValidator(goCtx context.Context, msg *types.MsgCreateVa
return nil, err
}

if msg.Commission.Rate.LT(k.MinCommissionRate(ctx)) {
return nil, sdkerrors.Wrapf(errors.New("commission cannot be less than min rate"), "cannot set validator commission to less than minimum rate of %s", k.MinCommissionRate(ctx))
}

// check to see if the pubkey or sender has been registered before
if _, found := k.GetValidator(ctx, valAddr); found {
return nil, types.ErrValidatorOwnerExists
Expand Down
Loading

0 comments on commit 6ebf1a5

Please sign in to comment.