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

V18 Upgrade Migrations #1084

Merged
merged 41 commits into from
Jan 26, 2024
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
dbfc0c3
re-fetched updated records after refresh
sampocs Jan 16, 2024
df83c77
added unit test for native refresh
sampocs Jan 17, 2024
910490d
Merge branch 'main' into reload-after-refresh
sampocs Jan 17, 2024
8354eaa
add upgrade handler
shellvish Jan 18, 2024
ff6fbab
Merge remote-tracking branch 'origin/reload-after-refresh' into v18-u…
shellvish Jan 18, 2024
c67c9f9
remove hub tokenizations
shellvish Jan 18, 2024
a3bd21c
migrate URRs
shellvish Jan 19, 2024
6552b59
set host zone unbonding at end of migration
shellvish Jan 19, 2024
a1773ab
nit changes
sampocs Jan 25, 2024
a66d8c4
Update x/stakeibc/keeper/unbonding_records.go
shellvish Jan 25, 2024
c3beefa
added constants file
sampocs Jan 25, 2024
421b1e6
switched to using redemption rate instead of unbond amount
sampocs Jan 25, 2024
9ac0359
Merge branch 'v18-upgrade-migrations' of github.com:Stride-Labs/strid…
sampocs Jan 25, 2024
02784bf
nit
sampocs Jan 25, 2024
ecfbd8b
nit
sampocs Jan 25, 2024
2759211
bugs
sampocs Jan 25, 2024
ebe7c3d
unit test
sampocs Jan 26, 2024
6403334
added starting epoch catch
sampocs Jan 26, 2024
3ae173c
Merge branch 'main' into v18-upgrade-migrations
asalzmann Jan 26, 2024
59d980a
add store key (#1088)
asalzmann Jan 26, 2024
72113ff
added upgrade unit test
sampocs Jan 26, 2024
98db45a
Merge branch 'v18-upgrade-migrations' of github.com:Stride-Labs/strid…
sampocs Jan 26, 2024
51eb82b
decrement terra delegations in progress by 3 in upgrade (#1089)
riley-stride Jan 26, 2024
3d9e44d
fixed unbonding unit tests (#1090)
sampocs Jan 26, 2024
f5ec490
added test case for status
sampocs Jan 26, 2024
ad69282
Merge branch 'v18-upgrade-migrations' of github.com:Stride-Labs/strid…
sampocs Jan 26, 2024
78b0e5f
lint fix
sampocs Jan 26, 2024
a73128a
added additional upgrade unit tests
sampocs Jan 26, 2024
ea02062
nit
sampocs Jan 26, 2024
289e922
nit
sampocs Jan 26, 2024
2f2f290
update constants
shellvish Jan 26, 2024
64fa722
Revert "update constants"
shellvish Jan 26, 2024
86c642d
Merge branch 'v18-upgrade-migrations' of github.com:Stride-Labs/strid…
sampocs Jan 26, 2024
bc3ec3d
Update constants.go
shellvish Jan 26, 2024
7f36d24
update starting estimate
shellvish Jan 26, 2024
1f06083
Update constants.go
shellvish Jan 26, 2024
5912787
Signaling prop in upgrade (#1092)
sampocs Jan 26, 2024
f9dea85
update constants with most recent data
shellvish Jan 26, 2024
0dd8f75
Merge branch 'v18-upgrade-migrations' of github.com:Stride-Labs/strid…
sampocs Jan 26, 2024
011a78a
fixed unit test after refresh
sampocs Jan 26, 2024
56602ca
Merge branch 'main' into v18-upgrade-migrations
sampocs Jan 26, 2024
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
12 changes: 12 additions & 0 deletions app/upgrades.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import (
v15 "github.com/Stride-Labs/stride/v17/app/upgrades/v15"
v16 "github.com/Stride-Labs/stride/v17/app/upgrades/v16"
v17 "github.com/Stride-Labs/stride/v17/app/upgrades/v17"
v18 "github.com/Stride-Labs/stride/v17/app/upgrades/v18"
v2 "github.com/Stride-Labs/stride/v17/app/upgrades/v2"
v3 "github.com/Stride-Labs/stride/v17/app/upgrades/v3"
v4 "github.com/Stride-Labs/stride/v17/app/upgrades/v4"
Expand Down Expand Up @@ -232,6 +233,17 @@ func (app *StrideApp) setupUpgradeHandlers(appOpts servertypes.AppOptions) {
),
)

// v18 upgrade handler
app.UpgradeKeeper.SetUpgradeHandler(
v18.UpgradeName,
v18.CreateUpgradeHandler(
app.mm,
app.configurator,
app.StakeibcKeeper,
app.RecordsKeeper,
),
)

upgradeInfo, err := app.UpgradeKeeper.ReadUpgradeInfoFromDisk()
if err != nil {
panic(fmt.Errorf("Failed to read upgrade info from disk: %w", err))
Expand Down
222 changes: 222 additions & 0 deletions app/upgrades/v18/upgrades.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,222 @@
package v18

import (
"fmt"
"strconv"
"strings"

errorsmod "cosmossdk.io/errors"
sdkmath "cosmossdk.io/math"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/types/module"
upgradetypes "github.com/cosmos/cosmos-sdk/x/upgrade/types"

recordskeeper "github.com/Stride-Labs/stride/v17/x/records/keeper"
recordtypes "github.com/Stride-Labs/stride/v17/x/records/types"
stakeibckeeper "github.com/Stride-Labs/stride/v17/x/stakeibc/keeper"
stakeibctypes "github.com/Stride-Labs/stride/v17/x/stakeibc/types"
)

func NewIntFromString(val string) sdkmath.Int {
out_int, ok := sdkmath.NewIntFromString(val)
sampocs marked this conversation as resolved.
Show resolved Hide resolved
if !ok {
panic(fmt.Sprintf("invalid string %s", val))
}
return out_int
}

var (
UpgradeName = "v18"

// Redemption rate bounds updated to give ~3 months of slack on outer bounds
RedemptionRateOuterMinAdjustment = sdk.MustNewDecFromStr("0.05")
RedemptionRateOuterMaxAdjustment = sdk.MustNewDecFromStr("0.10")

// Osmosis will have a slighly larger buffer with the redemption rate
// since their yield is less predictable
OsmosisChainId = "osmosis-1"
OsmosisRedemptionRateBuffer = sdk.MustNewDecFromStr("0.02")

// Get Initial Redemption Rates for Unbonding Records Migration
InitialRedemptionRates = map[string]sdk.Dec{
"comdex-1": sdk.MustNewDecFromStr("1.199107579864670291"),
sampocs marked this conversation as resolved.
Show resolved Hide resolved
"cosmoshub-4": sdk.MustNewDecFromStr("1.295759028526771630"),
"evmos_9001-2": sdk.MustNewDecFromStr("1.489961582552824897"),
"injective-1": sdk.MustNewDecFromStr("1.211514556746897031"),
"juno-1": sdk.MustNewDecFromStr("1.414898221278138305"),
"osmosis-1": sdk.MustNewDecFromStr("1.199042531372783387"),
"phoenix-1": sdk.MustNewDecFromStr("1.175092793361528752"),
"sommelier-3": sdk.MustNewDecFromStr("1.025191976926387110"),
"stargaze-1": sdk.MustNewDecFromStr("1.426572776580499529"),
"umee-1": sdk.MustNewDecFromStr("1.125555216159213353"),
}

// Get Amount Unbonded for each HostZone for Unbonding Records Migration
TotalAmountUnbonded = map[string]map[string]sdkmath.Int{
"comdex-1": {
"1705863627": NewIntFromString("1640812802"),
sampocs marked this conversation as resolved.
Show resolved Hide resolved
"1706209223": NewIntFromString("164530181"),
"1706554815": NewIntFromString("966328519777"),
"1706900422": NewIntFromString("334284598402"),
"1707246012": NewIntFromString("78876482507"),
},
"cosmoshub-4": {
"1706209219": NewIntFromString("495520981916"),
"1707246028": NewIntFromString("153692215135"),
"1706900433": NewIntFromString("101590127858"),
"1706554814": NewIntFromString("221083458468"),
"1705863634": NewIntFromString("142964081509"),
},
"evmos_9001-2": {
"1705690816": NewIntFromString("420455301467019532445082"),
"1705950016": NewIntFromString("1056499615224518202135947"),
"1706209220": NewIntFromString("149337601388876187205598"),
"1706468419": NewIntFromString("97606478355885814008907"),
"1706727614": NewIntFromString("1214076305566194153808"),
},
"injective-1": {
"1707246015": NewIntFromString("51607643878504534441739"),
},
"juno-1": {
"1706554816": NewIntFromString("157918257"),
"1706986817": NewIntFromString("176556452"),
"1707850813": NewIntFromString("447906130"),
"1706122818": NewIntFromString("4784225328"),
"1707418816": NewIntFromString("11456373"),
"1705690818": NewIntFromString("32566654"),
},
"osmosis-1": {
"1705690839": NewIntFromString("69534834841"),
"1705950052": NewIntFromString("34185116581"),
"1706209219": NewIntFromString("31400303772"),
"1706468412": NewIntFromString("58456099905"),
"1706727627": NewIntFromString("29541728198"),
},
"phoenix-1": {
"1706554818": NewIntFromString("14560879404"),
"1706900423": NewIntFromString("1576394459"),
"1707246013": NewIntFromString("1060350649"),
},
"sommelier-3": {
"1705690818": NewIntFromString("14786162726"),
"1706122819": NewIntFromString("3634"),
"1706554818": NewIntFromString("19330705153"),
"1706986821": NewIntFromString("21799461066"),
"1707418817": NewIntFromString("140098739"),
"1707850817": NewIntFromString("7879600093"),
},
"stargaze-1": {
"1705690816": NewIntFromString("327679537357"),
"1705950009": NewIntFromString("437989561516"),
"1706209215": NewIntFromString("437233672994"),
"1706468412": NewIntFromString("508229473185"),
"1706727610": NewIntFromString("1033532361858"),
},
"umee-1": {
"1705690815": NewIntFromString("510084698639"),
},
}
)

// CreateUpgradeHandler creates an SDK upgrade handler for v18
func CreateUpgradeHandler(
mm *module.Manager,
configurator module.Configurator,
stakeibcKeeper stakeibckeeper.Keeper,
recordsKeeper recordskeeper.Keeper,
) upgradetypes.UpgradeHandler {
return func(ctx sdk.Context, _ upgradetypes.Plan, vm module.VersionMap) (module.VersionMap, error) {
ctx.Logger().Info("Starting upgrade v18...")

ctx.Logger().Info("Updating redemption rate bounds...")
UpdateRedemptionRateBounds(ctx, stakeibcKeeper)

ctx.Logger().Info("Updating unbonding records...")
err := UpdateUnbondingRecords(ctx, stakeibcKeeper, recordsKeeper)
if err != nil {
return vm, err
}

return mm.RunMigrations(ctx, configurator, vm)
}
}

// Updates the outer redemption rate bounds
func UpdateRedemptionRateBounds(ctx sdk.Context, k stakeibckeeper.Keeper) {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
for _, hostZone := range k.GetAllHostZone(ctx) {
// Give osmosis a bit more slack since OSMO stakers collect real yield
outerAdjustment := RedemptionRateOuterMaxAdjustment
if hostZone.ChainId == OsmosisChainId {
outerAdjustment = outerAdjustment.Add(OsmosisRedemptionRateBuffer)
}

outerMinDelta := hostZone.RedemptionRate.Mul(RedemptionRateOuterMinAdjustment)
outerMaxDelta := hostZone.RedemptionRate.Mul(outerAdjustment)

outerMin := hostZone.RedemptionRate.Sub(outerMinDelta)
outerMax := hostZone.RedemptionRate.Add(outerMaxDelta)

hostZone.MinRedemptionRate = outerMin
hostZone.MaxRedemptionRate = outerMax

k.SetHostZone(ctx, hostZone)
}
}

// Modify HostZoneUnbonding and UserRedemptionRecords NativeTokenAmount to reflect new data structs
func UpdateUnbondingRecords(ctx sdk.Context, sk stakeibckeeper.Keeper, rk recordskeeper.Keeper) error {
// loop over host zone unbonding records
for _, epochUnbondingRecord := range rk.GetAllEpochUnbondingRecord(ctx) {
for _, hostZoneUnbonding := range epochUnbondingRecord.HostZoneUnbondings {

// we can ignore any record that's not currently unbonding
if hostZoneUnbonding.Status != recordtypes.HostZoneUnbonding_EXIT_TRANSFER_QUEUE {
continue
}
amountUnbonded := sdkmath.ZeroInt()

// check if the unbond amount is stored in cache above
allAmountsUnbondedForHost, found := TotalAmountUnbonded[hostZoneUnbonding.HostZoneId]
if !found {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
// check if this unbond time is in the cache above
unbondTime := strconv.FormatUint(hostZoneUnbonding.UnbondingTime, 10)
for unbondTimePrefixInCache, amountUnbondedInCache := range allAmountsUnbondedForHost {
if strings.HasPrefix(unbondTime, unbondTimePrefixInCache) {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
amountUnbonded = amountUnbondedInCache
break
}
}
}

// if we can't find unbonding amount, then we use an estimated redemption rate
if amountUnbonded.IsZero() {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
sampocs marked this conversation as resolved.
Show resolved Hide resolved
initialRedemptionRate := InitialRedemptionRates[hostZoneUnbonding.HostZoneId]
hostZone, found := sk.GetHostZone(ctx, hostZoneUnbonding.HostZoneId)
if !found {
return errorsmod.Wrapf(stakeibctypes.ErrHostZoneNotFound, "unable to find host zone with chain-id %s", hostZoneUnbonding.HostZoneId)
}
currentRedemptionRate := hostZone.RedemptionRate
blendedRedemptionRate := initialRedemptionRate.Add(currentRedemptionRate).Quo(sdk.NewDecFromInt(sdk.NewInt(2)))
sampocs marked this conversation as resolved.
Show resolved Hide resolved
amountUnbonded = sdk.NewDecFromInt(hostZoneUnbonding.StTokenAmount).Mul(blendedRedemptionRate).TruncateInt()
}
hostZoneUnbonding.NativeTokenAmount = amountUnbonded
sampocs marked this conversation as resolved.
Show resolved Hide resolved

// get implied redemption rate, based on the amount unbonded and stTokens burned
impliedRedemptionRate := sdk.NewDecFromInt(amountUnbonded).Quo(sdk.NewDecFromInt(hostZoneUnbonding.StTokenAmount))

// now update all userRedemptionRecords
for _, userRedemptionRecordId := range hostZoneUnbonding.UserRedemptionRecords {
asalzmann marked this conversation as resolved.
Show resolved Hide resolved
userRedemptionRecord, found := rk.GetUserRedemptionRecord(ctx, userRedemptionRecordId)
if !found {
return errorsmod.Wrapf(recordtypes.ErrHostUnbondingRecordNotFound, "unable to find user redemption record with id %s", userRedemptionRecordId)
}
userRedemptionRecord.NativeTokenAmount = sdk.NewDecFromInt(userRedemptionRecord.StTokenAmount).Mul(impliedRedemptionRate).TruncateInt()
rk.SetUserRedemptionRecord(ctx, userRedemptionRecord)
}

sampocs marked this conversation as resolved.
Show resolved Hide resolved
// finally, update the hostZoneUnbonding record
return rk.SetHostZoneUnbondingRecord(ctx, epochUnbondingRecord.EpochNumber, hostZoneUnbonding.HostZoneId, *hostZoneUnbonding)
}
}
return nil
}
5 changes: 0 additions & 5 deletions x/stakeibc/keeper/hooks.go
Original file line number Diff line number Diff line change
Expand Up @@ -101,11 +101,6 @@ func (k Keeper) BeforeEpochStart(ctx sdk.Context, epochInfo epochstypes.EpochInf
k.SwapAllRewardTokens(ctx)
}

// TODO [cleanup]: Remove after v17 upgrade
// Submit ICA to disable gaia tokenization (this only needs to be run once)
if epochInfo.Identifier == epochstypes.DAY_EPOCH && epochNumber%10 == 0 {
k.DisableHubTokenization(ctx)
}
}

func (k Keeper) AfterEpochEnd(ctx sdk.Context, epochInfo epochstypes.EpochInfo) {}
Expand Down
7 changes: 5 additions & 2 deletions x/stakeibc/keeper/unbonding_records.go
Original file line number Diff line number Diff line change
Expand Up @@ -441,14 +441,17 @@
}

// Get the list of relevant records that should unbond
epochUnbondingRecordIds, epochNumberToHostZoneUnbondingMap := k.GetQueuedHostZoneUnbondingRecords(ctx, hostZone.ChainId)
_, initialEpochNumberToHostZoneUnbondingMap := k.GetQueuedHostZoneUnbondingRecords(ctx, hostZone.ChainId)

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride amd64 for darwin

initialEpochNumberToHostZoneUnbondingMap declared and not used

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

initialEpochNumberToHostZoneUnbondingMap declared and not used

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

initialEpochNumberToHostZoneUnbondingMap declared and not used

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

initialEpochNumberToHostZoneUnbondingMap declared and not used

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride amd64 for linux

initialEpochNumberToHostZoneUnbondingMap declared and not used

Check failure on line 444 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride arm64 for darwin

initialEpochNumberToHostZoneUnbondingMap declared and not used

// Update the native unbond amount on all relevant records
// The native amount is calculated from the stTokens
if err := k.RefreshUnbondingNativeTokenAmounts(ctx, epochNumberToHostZoneUnbondingMap); err != nil {
if err := k.RefreshUnbondingNativeTokenAmounts(ctx, iniitalEpochNumberToHostZoneUnbondingMap); err != nil {

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride amd64 for darwin

undefined: iniitalEpochNumberToHostZoneUnbondingMap

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

undefined: iniitalEpochNumberToHostZoneUnbondingMap (typecheck)

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

undefined: iniitalEpochNumberToHostZoneUnbondingMap) (typecheck)

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / golangci-lint

undefined: iniitalEpochNumberToHostZoneUnbondingMap) (typecheck)

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride amd64 for linux

undefined: iniitalEpochNumberToHostZoneUnbondingMap

Check failure on line 448 in x/stakeibc/keeper/unbonding_records.go

View workflow job for this annotation

GitHub Actions / stride arm64 for darwin

undefined: iniitalEpochNumberToHostZoneUnbondingMap
return err
}

// Fetch the records again with the updated native amounts
epochUnbondingRecordIds, epochNumberToHostZoneUnbondingMap := k.GetQueuedHostZoneUnbondingRecords(ctx, hostZone.ChainId)

// Sum the total number of native tokens that from the records above that are ready to unbond
totalUnbondAmount := k.GetTotalUnbondAmount(ctx, epochNumberToHostZoneUnbondingMap)
k.Logger(ctx).Info(utils.LogWithHostZone(hostZone.ChainId,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
package keeper_test

import (
"fmt"

sdkmath "cosmossdk.io/math"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/gogoproto/proto"
Expand Down Expand Up @@ -58,22 +60,34 @@ func (s *KeeperTestSuite) SetupTestUnbondFromHostZone(
DelegationIcaAddress: "cosmos_DELEGATION",
Validators: validators,
TotalDelegations: totalStake,
RedemptionRate: sdk.OneDec(),
}
s.App.StakeibcKeeper.SetHostZone(s.Ctx, hostZone)

// Store the total unbond amount across two epoch unbonding records
// and create a user redemption record for each
halfUnbondAmount := unbondAmount.Quo(sdkmath.NewInt(2))
for i := uint64(1); i <= 2; i++ {
redemptionRecordId := fmt.Sprintf("id-%d", i)

s.App.RecordsKeeper.SetEpochUnbondingRecord(s.Ctx, recordtypes.EpochUnbondingRecord{
EpochNumber: i,
HostZoneUnbondings: []*recordtypes.HostZoneUnbonding{
{
HostZoneId: HostChainId,
Status: recordtypes.HostZoneUnbonding_UNBONDING_QUEUE,
NativeTokenAmount: halfUnbondAmount,
HostZoneId: HostChainId,
Status: recordtypes.HostZoneUnbonding_UNBONDING_QUEUE,
StTokenAmount: halfUnbondAmount,
NativeTokenAmount: halfUnbondAmount,
UserRedemptionRecords: []string{redemptionRecordId},
},
},
})

s.App.RecordsKeeper.SetUserRedemptionRecord(s.Ctx, recordtypes.UserRedemptionRecord{
Id: redemptionRecordId,
StTokenAmount: halfUnbondAmount,
NativeTokenAmount: halfUnbondAmount,
})
}

// Mock the epoch tracker to timeout 90% through the epoch
Expand Down Expand Up @@ -351,6 +365,42 @@ func (s *KeeperTestSuite) TestUnbondFromHostZone_Successful_UnbondTotalGreaterTh
s.CheckUnbondingMessages(tc, expectedUnbondings)
}

func (s *KeeperTestSuite) TestUnbondFromHostZone_Successful_RefreshedNativeAmount() {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
// Total Stake: 1000
//
// Unbond Amount with old redemption rate (RR = 1): 100
// Unbond Amount with new redemption rate (RR = 1.5): 150
//
// Stake After Unbond: 850
updatedRedemptionRate := sdk.MustNewDecFromStr("1.5")
unbondAmountWithOldRate := sdkmath.NewInt(100)
unbondAmountWithNewRate := sdkmath.NewInt(150)
totalStake := sdkmath.NewInt(1000)
totalWeight := int64(100)

// Since this test is only intended to check the native token refresh,
// we don't need more than 1 validator
// That validator should unbond the full amount with the new redemption rate
validators := []*types.Validator{
{Address: "valA", Weight: 100, Delegation: totalStake},
}
expectedUnbondings := []ValidatorUnbonding{
{Validator: "valA", UnbondAmount: unbondAmountWithNewRate},
}

// Setup using default, and then override the redemption rate value to update it from 1.0 to 1.5
tc := s.SetupTestUnbondFromHostZone(totalWeight, totalStake, unbondAmountWithOldRate, validators)

hostZone := s.MustGetHostZone(HostChainId)
hostZone.RedemptionRate = updatedRedemptionRate
s.App.StakeibcKeeper.SetHostZone(s.Ctx, hostZone)

// Finally check that the unbondings matched - mostly checking that there was a greater amount
// unbonded than was originally in the host zone unbonding record
tc.totalUnbondAmount = unbondAmountWithNewRate
s.CheckUnbondingMessages(tc, expectedUnbondings)
}

func (s *KeeperTestSuite) TestUnbondFromHostZone_NoDelegationAccount() {
// Call unbond on a host zone without a delegation account - it should error
invalidHostZone := types.HostZone{
Expand Down
Loading