From 90686e0eafa9ccc4a7e2cb311e2c5a6b203ec7ec Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 28 Apr 2023 17:59:55 -0300 Subject: [PATCH 01/24] feat(errors): Add ErrStopIterating --- errors/errors.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/errors/errors.go b/errors/errors.go index 088476c81d82..2e5140d72828 100644 --- a/errors/errors.go +++ b/errors/errors.go @@ -16,6 +16,9 @@ var ( // errInternal should never be exposed, but we reserve this code for non-specified errors errInternal = Register(UndefinedCodespace, 1, "internal") + // ErrStopIterating is used to break out of an iteration + ErrStopIterating = Register(UndefinedCodespace, 2, "stop iterating") + // ErrPanic should only be set when we recovering from a panic ErrPanic = Register(UndefinedCodespace, 111222, "panic") ) From ee2d278a7d118acf4c707ced8f668a90e9db08ea Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 00:29:14 +0200 Subject: [PATCH 02/24] progress --- x/upgrade/abci.go | 55 +++-- x/upgrade/abci_test.go | 59 +++--- x/upgrade/keeper/grpc_query.go | 32 ++- x/upgrade/keeper/grpc_query_test.go | 13 +- x/upgrade/keeper/keeper.go | 299 ++++++++++++++++++---------- x/upgrade/keeper/keeper_test.go | 62 +++--- x/upgrade/keeper/migrations.go | 18 +- x/upgrade/keeper/migrations_test.go | 17 +- x/upgrade/keeper/msg_server_test.go | 8 +- x/upgrade/module.go | 22 +- x/upgrade/types/handler.go | 5 +- x/upgrade/types/plan.go | 10 +- x/upgrade/types/plan_test.go | 6 +- 13 files changed, 385 insertions(+), 221 deletions(-) diff --git a/x/upgrade/abci.go b/x/upgrade/abci.go index ae72c678edf2..b0911e24e07f 100644 --- a/x/upgrade/abci.go +++ b/x/upgrade/abci.go @@ -1,6 +1,7 @@ package upgrade import ( + "context" "fmt" "time" @@ -8,8 +9,9 @@ import ( "cosmossdk.io/x/upgrade/keeper" "cosmossdk.io/x/upgrade/types" - "github.com/cosmos/cosmos-sdk/telemetry" sdk "github.com/cosmos/cosmos-sdk/types" + + "github.com/cosmos/cosmos-sdk/telemetry" ) // BeginBlock will check if there is a scheduled plan and if it is ready to be executed. @@ -20,10 +22,15 @@ import ( // The purpose is to ensure the binary is switched EXACTLY at the desired block, and to allow // a migration to be executed if needed upon this switch (migration defined in the new binary) // skipUpgradeHeightArray is a set of block heights for which the upgrade must be skipped -func BeginBlocker(k *keeper.Keeper, ctx sdk.Context) { +func BeginBlocker(k *keeper.Keeper, ctx context.Context) error { defer telemetry.ModuleMeasureSince(types.ModuleName, time.Now(), telemetry.MetricKeyBeginBlocker) - plan, found := k.GetUpgradePlan(ctx) + sdkCtx := sdk.UnwrapSDKContext(ctx) + blockHeight := sdkCtx.BlockHeight() + plan, found, err := k.GetUpgradePlan(ctx) + if err != nil { + return err + } if !k.DowngradeVerified() { k.SetDowngradeVerified(true) @@ -32,66 +39,74 @@ func BeginBlocker(k *keeper.Keeper, ctx sdk.Context) { // 1. If there is no scheduled upgrade. // 2. If the plan is not ready. // 3. If the plan is ready and skip upgrade height is set for current height. - if !found || !plan.ShouldExecute(ctx) || (plan.ShouldExecute(ctx) && k.IsSkipHeight(ctx.BlockHeight())) { - lastAppliedPlan, _ := k.GetLastCompletedUpgrade(ctx) + if !found || !plan.ShouldExecute(blockHeight) || (plan.ShouldExecute(blockHeight) && k.IsSkipHeight(blockHeight)) { + lastAppliedPlan, _, err := k.GetLastCompletedUpgrade(ctx) + if err != nil { + return err + } + if lastAppliedPlan != "" && !k.HasHandler(lastAppliedPlan) { var appVersion uint64 - cp := ctx.ConsensusParams() + cp := sdkCtx.ConsensusParams() if cp.Version != nil { appVersion = cp.Version.App } - panic(fmt.Sprintf("Wrong app version %d, upgrade handler is missing for %s upgrade plan", appVersion, lastAppliedPlan)) + return fmt.Errorf("Wrong app version %d, upgrade handler is missing for %s upgrade plan", appVersion, lastAppliedPlan) } } } if !found { - return + return nil } - logger := ctx.Logger() + + logger := k.Logger(ctx) // To make sure clear upgrade is executed at the same block - if plan.ShouldExecute(ctx) { + if plan.ShouldExecute(blockHeight) { // If skip upgrade has been set for current height, we clear the upgrade plan - if k.IsSkipHeight(ctx.BlockHeight()) { + if k.IsSkipHeight(blockHeight) { skipUpgradeMsg := fmt.Sprintf("UPGRADE \"%s\" SKIPPED at %d: %s", plan.Name, plan.Height, plan.Info) logger.Info(skipUpgradeMsg) // Clear the upgrade plan at current height - k.ClearUpgradePlan(ctx) - return + return k.ClearUpgradePlan(ctx) } // Prepare shutdown if we don't have an upgrade handler for this upgrade name (meaning this software is out of date) if !k.HasHandler(plan.Name) { // Write the upgrade info to disk. The UpgradeStoreLoader uses this info to perform or skip // store migrations. - err := k.DumpUpgradeInfoToDisk(ctx.BlockHeight(), plan) + err := k.DumpUpgradeInfoToDisk(blockHeight, plan) if err != nil { - panic(fmt.Errorf("unable to write upgrade info to filesystem: %s", err.Error())) + return fmt.Errorf("unable to write upgrade info to filesystem: %s", err.Error()) } upgradeMsg := BuildUpgradeNeededMsg(plan) logger.Error(upgradeMsg) + // TODO: figure out if I can convert this message into an error (will it panic the same way?) panic(upgradeMsg) } // We have an upgrade handler for this upgrade name, so apply the upgrade - ctx.Logger().Info(fmt.Sprintf("applying upgrade \"%s\" at %s", plan.Name, plan.DueAt())) - ctx = ctx.WithBlockGasMeter(storetypes.NewInfiniteGasMeter()) - k.ApplyUpgrade(ctx, plan) - return + logger.Info(fmt.Sprintf("applying upgrade \"%s\" at %s", plan.Name, plan.DueAt())) + sdkCtx = sdkCtx.WithBlockGasMeter(storetypes.NewInfiniteGasMeter()) + // TODO: figure out how to pass a block gas meter to the ApplyUpgradeFunction + return k.ApplyUpgrade(sdk.WrapSDKContext(sdkCtx), plan) } // if we have a pending upgrade, but it is not yet time, make sure we did not // set the handler already if k.HasHandler(plan.Name) { downgradeMsg := fmt.Sprintf("BINARY UPDATED BEFORE TRIGGER! UPGRADE \"%s\" - in binary but not executed on chain. Downgrade your binary", plan.Name) - ctx.Logger().Error(downgradeMsg) + logger.Error(downgradeMsg) + // TODO: figure out if I can convert this message into an error (will it panic the same way?) panic(downgradeMsg) } + + return nil } // BuildUpgradeNeededMsg prints the message that notifies that an upgrade is needed. diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index c3ccf4a0871a..be502360c47c 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -1,6 +1,7 @@ package upgrade_test import ( + "context" "errors" "os" "testing" @@ -10,8 +11,12 @@ import ( "cosmossdk.io/log" storetypes "cosmossdk.io/store/types" cmtproto "github.com/cometbft/cometbft/proto/tendermint/types" + "github.com/stretchr/testify/require" + "github.com/stretchr/testify/suite" + "github.com/cosmos/cosmos-sdk/baseapp" addresscodec "github.com/cosmos/cosmos-sdk/codec/address" + "github.com/cosmos/cosmos-sdk/runtime" "github.com/cosmos/cosmos-sdk/testutil" sdk "github.com/cosmos/cosmos-sdk/types" sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" @@ -20,8 +25,6 @@ import ( authtypes "github.com/cosmos/cosmos-sdk/x/auth/types" govtypes "github.com/cosmos/cosmos-sdk/x/gov/types" govtypesv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" - "github.com/stretchr/testify/require" - "github.com/stretchr/testify/suite" "cosmossdk.io/x/upgrade" "cosmossdk.io/x/upgrade/keeper" @@ -44,6 +47,7 @@ var s TestSuite func setupTest(t *testing.T, height int64, skip map[int64]bool) *TestSuite { s.encCfg = moduletestutil.MakeTestEncodingConfig(upgrade.AppModuleBasic{}) key := storetypes.NewKVStoreKey(types.StoreKey) + storeService := runtime.NewKVStoreService(key) testCtx := testutil.DefaultContextWithDB(s.T(), key, storetypes.NewTransientStoreKey("transient_test")) s.baseApp = baseapp.NewBaseApp( @@ -53,7 +57,7 @@ func setupTest(t *testing.T, height int64, skip map[int64]bool) *TestSuite { s.encCfg.TxConfig.TxDecoder(), ) - s.keeper = keeper.NewKeeper(skip, key, s.encCfg.Codec, t.TempDir(), nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + s.keeper = keeper.NewKeeper(skip, storeService, s.encCfg.Codec, t.TempDir(), nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) s.keeper.SetVersionSetter(s.baseApp) s.ctx = testCtx.Ctx.WithBlockHeader(cmtproto.Header{Time: time.Now(), Height: height}) @@ -107,7 +111,7 @@ func VerifyDoUpgrade(t *testing.T) { }) t.Log("Verify that the upgrade can be successfully applied with a handler") - s.keeper.SetUpgradeHandler("test", func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + s.keeper.SetUpgradeHandler("test", func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) require.NotPanics(t, func() { @@ -124,7 +128,7 @@ func VerifyDoUpgradeWithCtx(t *testing.T, newCtx sdk.Context, proposalName strin }) t.Log("Verify that the upgrade can be successfully applied with a handler") - s.keeper.SetUpgradeHandler(proposalName, func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + s.keeper.SetUpgradeHandler(proposalName, func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) require.NotPanics(t, func() { @@ -138,7 +142,7 @@ func TestHaltIfTooNew(t *testing.T) { s := setupTest(t, 10, map[int64]bool{}) t.Log("Verify that we don't panic with registered plan not in database at all") var called int - s.keeper.SetUpgradeHandler("future", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + s.keeper.SetUpgradeHandler("future", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { called++ return vm, nil }) @@ -170,9 +174,10 @@ func TestHaltIfTooNew(t *testing.T) { func VerifyCleared(t *testing.T, newCtx sdk.Context) { t.Log("Verify that the upgrade plan has been cleared") - plan, _ := s.keeper.GetUpgradePlan(newCtx) + plan, _, err := s.keeper.GetUpgradePlan(newCtx) expected := types.Plan{} require.Equal(t, plan, expected) + require.NoError(t, err) } func TestCanClear(t *testing.T) { @@ -214,14 +219,16 @@ func TestPlanStringer(t *testing.T) { func VerifyNotDone(t *testing.T, newCtx sdk.Context, name string) { t.Log("Verify that upgrade was not done") - height := s.keeper.GetDoneHeight(newCtx, name) + height, err := s.keeper.GetDoneHeight(newCtx, name) require.Zero(t, height) + require.NoError(t, err) } func VerifyDone(t *testing.T, newCtx sdk.Context, name string) { t.Log("Verify that the upgrade plan has been executed") - height := s.keeper.GetDoneHeight(newCtx, name) + height, err := s.keeper.GetDoneHeight(newCtx, name) require.NotZero(t, height) + require.NoError(t, err) } func VerifySet(t *testing.T, skipUpgradeHeights map[int64]bool) { @@ -421,7 +428,7 @@ func TestBinaryVersion(t *testing.T) { { "test not panic: upgrade handler is present for last applied upgrade", func() sdk.Context { - s.keeper.SetUpgradeHandler("test0", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + s.keeper.SetUpgradeHandler("test0", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) @@ -470,12 +477,13 @@ func TestDowngradeVerification(t *testing.T) { // for the two keepers. encCfg := moduletestutil.MakeTestEncodingConfig(upgrade.AppModuleBasic{}) key := storetypes.NewKVStoreKey(types.StoreKey) + storeService := runtime.NewKVStoreService(key) testCtx := testutil.DefaultContextWithDB(s.T(), key, storetypes.NewTransientStoreKey("transient_test")) ctx := testCtx.Ctx.WithBlockHeader(cmtproto.Header{Time: time.Now(), Height: 10}) skip := map[int64]bool{} tempDir := t.TempDir() - k := keeper.NewKeeper(skip, key, encCfg.Codec, tempDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + k := keeper.NewKeeper(skip, storeService, encCfg.Codec, tempDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) m := upgrade.NewAppModule(k, addresscodec.NewBech32Codec("cosmos")) handler := upgrade.NewSoftwareUpgradeProposalHandler(k) @@ -486,7 +494,7 @@ func TestDowngradeVerification(t *testing.T) { ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 1) // set the handler. - k.SetUpgradeHandler(planName, func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + k.SetUpgradeHandler(planName, func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) @@ -498,11 +506,11 @@ func TestDowngradeVerification(t *testing.T) { testCases := map[string]struct { preRun func(*keeper.Keeper, sdk.Context, string) - expectPanic bool + expectError bool }{ "valid binary": { preRun: func(k *keeper.Keeper, ctx sdk.Context, name string) { - k.SetUpgradeHandler(planName, func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + k.SetUpgradeHandler(planName, func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) }, @@ -513,10 +521,10 @@ func TestDowngradeVerification(t *testing.T) { err := handler(ctx, &types.SoftwareUpgradeProposal{Title: "test", Plan: types.Plan{Name: "another" + planName, Height: ctx.BlockHeight() + 1}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err, name) }, - expectPanic: true, + expectError: true, }, "downgrade without any active plan": { - expectPanic: true, + expectError: true, }, } @@ -524,29 +532,28 @@ func TestDowngradeVerification(t *testing.T) { ctx, _ := ctx.CacheContext() // downgrade. now keeper does not have the handler. - k := keeper.NewKeeper(skip, key, encCfg.Codec, tempDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + k := keeper.NewKeeper(skip, storeService, encCfg.Codec, tempDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) m := upgrade.NewAppModule(k, addresscodec.NewBech32Codec("cosmos")) // assertions - lastAppliedPlan, _ := k.GetLastCompletedUpgrade(ctx) + lastAppliedPlan, _, err := k.GetLastCompletedUpgrade(ctx) + require.NoError(t, err) require.Equal(t, planName, lastAppliedPlan) require.False(t, k.HasHandler(planName)) require.False(t, k.DowngradeVerified()) - _, found := k.GetUpgradePlan(ctx) + _, found, err := k.GetUpgradePlan(ctx) + require.NoError(t, err) require.False(t, found) if tc.preRun != nil { tc.preRun(k, ctx, name) } - if tc.expectPanic { - require.Panics(t, func() { - m.BeginBlock(ctx) - }, name) + err = m.BeginBlock(ctx) + if tc.expectError { + require.Error(t, err, name) } else { - require.NotPanics(t, func() { - m.BeginBlock(ctx) - }, name) + require.NoError(t, err, name) } } } diff --git a/x/upgrade/keeper/grpc_query.go b/x/upgrade/keeper/grpc_query.go index d0c30c36423e..6b324e586ee6 100644 --- a/x/upgrade/keeper/grpc_query.go +++ b/x/upgrade/keeper/grpc_query.go @@ -16,7 +16,11 @@ var _ types.QueryServer = Keeper{} func (k Keeper) CurrentPlan(c context.Context, req *types.QueryCurrentPlanRequest) (*types.QueryCurrentPlanResponse, error) { ctx := sdk.UnwrapSDKContext(c) - plan, found := k.GetUpgradePlan(ctx) + plan, found, err := k.GetUpgradePlan(ctx) + if err != nil { + return nil, err + } + if !found { return &types.QueryCurrentPlanResponse{}, nil } @@ -28,19 +32,20 @@ func (k Keeper) CurrentPlan(c context.Context, req *types.QueryCurrentPlanReques func (k Keeper) AppliedPlan(c context.Context, req *types.QueryAppliedPlanRequest) (*types.QueryAppliedPlanResponse, error) { ctx := sdk.UnwrapSDKContext(c) - applied := k.GetDoneHeight(ctx, req.Name) - if applied == 0 { - return &types.QueryAppliedPlanResponse{}, nil - } + applied, err := k.GetDoneHeight(ctx, req.Name) - return &types.QueryAppliedPlanResponse{Height: applied}, nil + return &types.QueryAppliedPlanResponse{Height: applied}, err } // UpgradedConsensusState implements the Query/UpgradedConsensusState gRPC method func (k Keeper) UpgradedConsensusState(c context.Context, req *types.QueryUpgradedConsensusStateRequest) (*types.QueryUpgradedConsensusStateResponse, error) { //nolint:staticcheck // we're using a deprecated call for compatibility ctx := sdk.UnwrapSDKContext(c) - consState, found := k.GetUpgradedConsensusState(ctx, req.LastHeight) + consState, found, err := k.GetUpgradedConsensusState(ctx, req.LastHeight) + if err != nil { + return nil, err + } + if !found { return &types.QueryUpgradedConsensusStateResponse{}, nil //nolint:staticcheck // we're using a deprecated call for compatibility } @@ -56,7 +61,12 @@ func (k Keeper) ModuleVersions(c context.Context, req *types.QueryModuleVersions // check if a specific module was requested if len(req.ModuleName) > 0 { - if version, ok := k.getModuleVersion(ctx, req.ModuleName); ok { + version, ok, err := k.getModuleVersion(ctx, req.ModuleName) + if err != nil { + return nil, err + } + + if ok { // return the requested module res := []*types.ModuleVersion{{Name: req.ModuleName, Version: version}} return &types.QueryModuleVersionsResponse{ModuleVersions: res}, nil @@ -66,7 +76,11 @@ func (k Keeper) ModuleVersions(c context.Context, req *types.QueryModuleVersions } // if no module requested return all module versions from state - mv := k.GetModuleVersions(ctx) + mv, err := k.GetModuleVersions(ctx) + if err != nil { + return nil, err + } + return &types.QueryModuleVersionsResponse{ ModuleVersions: mv, }, nil diff --git a/x/upgrade/keeper/grpc_query_test.go b/x/upgrade/keeper/grpc_query_test.go index 61c628a40d22..eaf78790a85e 100644 --- a/x/upgrade/keeper/grpc_query_test.go +++ b/x/upgrade/keeper/grpc_query_test.go @@ -13,6 +13,7 @@ import ( "cosmossdk.io/x/upgrade/types" "github.com/cosmos/cosmos-sdk/baseapp" + "github.com/cosmos/cosmos-sdk/runtime" "github.com/cosmos/cosmos-sdk/testutil" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/module" @@ -33,12 +34,13 @@ type UpgradeTestSuite struct { func (suite *UpgradeTestSuite) SetupTest() { suite.encCfg = moduletestutil.MakeTestEncodingConfig(upgrade.AppModuleBasic{}) key := storetypes.NewKVStoreKey(types.StoreKey) + storeService := runtime.NewKVStoreService(key) testCtx := testutil.DefaultContextWithDB(suite.T(), key, storetypes.NewTransientStoreKey("transient_test")) suite.ctx = testCtx.Ctx skipUpgradeHeights := make(map[int64]bool) - suite.upgradeKeeper = keeper.NewKeeper(skipUpgradeHeights, key, suite.encCfg.Codec, "", nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + suite.upgradeKeeper = keeper.NewKeeper(skipUpgradeHeights, storeService, suite.encCfg.Codec, "", nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) suite.upgradeKeeper.SetModuleVersionMap(suite.ctx, module.VersionMap{ "bank": 0, }) @@ -127,7 +129,7 @@ func (suite *UpgradeTestSuite) TestAppliedCurrentPlan() { suite.upgradeKeeper.ScheduleUpgrade(suite.ctx, plan) suite.ctx = suite.ctx.WithBlockHeight(expHeight) - suite.upgradeKeeper.SetUpgradeHandler(planName, func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + suite.upgradeKeeper.SetUpgradeHandler(planName, func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) suite.upgradeKeeper.ApplyUpgrade(suite.ctx, plan) @@ -184,8 +186,11 @@ func (suite *UpgradeTestSuite) TestModuleVersions() { }, } - vm := suite.upgradeKeeper.GetModuleVersionMap(suite.ctx) - mv := suite.upgradeKeeper.GetModuleVersions(suite.ctx) + vm, err := suite.upgradeKeeper.GetModuleVersionMap(suite.ctx) + suite.Require().NoError(err) + + mv, err := suite.upgradeKeeper.GetModuleVersions(suite.ctx) + suite.Require().NoError(err) for _, tc := range testCases { suite.Run(fmt.Sprintf("Case %s", tc.msg), func() { diff --git a/x/upgrade/keeper/keeper.go b/x/upgrade/keeper/keeper.go index 143f8225fd5e..2dc150b1ef83 100644 --- a/x/upgrade/keeper/keeper.go +++ b/x/upgrade/keeper/keeper.go @@ -1,6 +1,7 @@ package keeper import ( + "context" "encoding/binary" "encoding/json" "fmt" @@ -10,6 +11,7 @@ import ( "sort" "strconv" + corestore "cosmossdk.io/core/store" errorsmod "cosmossdk.io/errors" "cosmossdk.io/log" "cosmossdk.io/store/prefix" @@ -18,7 +20,9 @@ import ( "cosmossdk.io/x/upgrade/types" "github.com/armon/go-metrics" + "github.com/cosmos/cosmos-sdk/codec" + "github.com/cosmos/cosmos-sdk/runtime" "github.com/cosmos/cosmos-sdk/telemetry" sdk "github.com/cosmos/cosmos-sdk/types" sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" @@ -33,7 +37,7 @@ const UpgradeInfoFileName string = "upgrade-info.json" type Keeper struct { homePath string // root directory of app config skipUpgradeHeights map[int64]bool // map of heights to skip for an upgrade - storeKey storetypes.StoreKey // key to access x/upgrade store + storeService corestore.KVStoreService // key to access x/upgrade store cdc codec.BinaryCodec // App-wide binary codec upgradeHandlers map[string]types.UpgradeHandler // map of plan name to upgrade handler versionSetter xp.ProtocolVersionSetter // implements setting the protocol version field on BaseApp @@ -48,11 +52,11 @@ type Keeper struct { // cdc - the app-wide binary codec // homePath - root directory of the application's config // vs - the interface implemented by baseapp which allows setting baseapp's protocol version field -func NewKeeper(skipUpgradeHeights map[int64]bool, storeKey storetypes.StoreKey, cdc codec.BinaryCodec, homePath string, vs xp.ProtocolVersionSetter, authority string) *Keeper { +func NewKeeper(skipUpgradeHeights map[int64]bool, storeService corestore.KVStoreService, cdc codec.BinaryCodec, homePath string, vs xp.ProtocolVersionSetter, authority string) *Keeper { k := &Keeper{ homePath: homePath, skipUpgradeHeights: skipUpgradeHeights, - storeKey: storeKey, + storeService: storeService, cdc: cdc, upgradeHandlers: map[string]types.UpgradeHandler{}, versionSetter: vs, @@ -96,31 +100,38 @@ func (k Keeper) SetUpgradeHandler(name string, upgradeHandler types.UpgradeHandl } // setProtocolVersion sets the protocol version to state -func (k Keeper) setProtocolVersion(ctx sdk.Context, v uint64) { - store := ctx.KVStore(k.storeKey) +func (k Keeper) setProtocolVersion(ctx context.Context, v uint64) error { + store := k.storeService.OpenKVStore(ctx) versionBytes := make([]byte, 8) binary.BigEndian.PutUint64(versionBytes, v) - store.Set([]byte{types.ProtocolVersionByte}, versionBytes) + return store.Set([]byte{types.ProtocolVersionByte}, versionBytes) } // getProtocolVersion gets the protocol version from state -func (k Keeper) getProtocolVersion(ctx sdk.Context) uint64 { - store := ctx.KVStore(k.storeKey) - ok := store.Has([]byte{types.ProtocolVersionByte}) +func (k Keeper) getProtocolVersion(ctx context.Context) (uint64, error) { + store := k.storeService.OpenKVStore(ctx) + ok, err := store.Has([]byte{types.ProtocolVersionByte}) + if err != nil { + return 0, err + } + if ok { - pvBytes := store.Get([]byte{types.ProtocolVersionByte}) - protocolVersion := binary.BigEndian.Uint64(pvBytes) + pvBytes, err := store.Get([]byte{types.ProtocolVersionByte}) + if err != nil { + return 0, err + } - return protocolVersion + protocolVersion := binary.BigEndian.Uint64(pvBytes) + return protocolVersion, nil } // default value - return 0 + return 0, nil } // SetModuleVersionMap saves a given version map to state -func (k Keeper) SetModuleVersionMap(ctx sdk.Context, vm module.VersionMap) { +func (k Keeper) SetModuleVersionMap(ctx context.Context, vm module.VersionMap) error { if len(vm) > 0 { - store := ctx.KVStore(k.storeKey) + store := runtime.KVStoreAdapter(k.storeService.OpenKVStore(ctx)) versionStore := prefix.NewStore(store, []byte{types.VersionMapByte}) // Even though the underlying store (cachekv) store is sorted, we still // prefer a deterministic iteration order of the map, to avoid undesired @@ -140,16 +151,21 @@ func (k Keeper) SetModuleVersionMap(ctx sdk.Context, vm module.VersionMap) { versionStore.Set(nameBytes, verBytes) } } + + return nil } // GetModuleVersionMap returns a map of key module name and value module consensus version // as defined in ADR-041. -func (k Keeper) GetModuleVersionMap(ctx sdk.Context) module.VersionMap { - store := ctx.KVStore(k.storeKey) - it := storetypes.KVStorePrefixIterator(store, []byte{types.VersionMapByte}) +func (k Keeper) GetModuleVersionMap(ctx context.Context) (module.VersionMap, error) { + store := k.storeService.OpenKVStore(ctx) + prefix := []byte{types.VersionMapByte} + it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) + if err != nil { + return nil, err + } vm := make(module.VersionMap) - defer it.Close() for ; it.Valid(); it.Next() { moduleBytes := it.Key() // first byte is prefix key, so we remove it here @@ -158,14 +174,17 @@ func (k Keeper) GetModuleVersionMap(ctx sdk.Context) module.VersionMap { vm[name] = moduleVersion } - return vm + return vm, it.Close() } // GetModuleVersions gets a slice of module consensus versions -func (k Keeper) GetModuleVersions(ctx sdk.Context) []*types.ModuleVersion { - store := ctx.KVStore(k.storeKey) - it := storetypes.KVStorePrefixIterator(store, []byte{types.VersionMapByte}) - defer it.Close() +func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, error) { + store := k.storeService.OpenKVStore(ctx) + prefix := []byte{types.VersionMapByte} + it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) + if err != nil { + return nil, err + } mv := make([]*types.ModuleVersion, 0) for ; it.Valid(); it.Next() { @@ -177,54 +196,79 @@ func (k Keeper) GetModuleVersions(ctx sdk.Context) []*types.ModuleVersion { Version: moduleVersion, }) } - return mv + + return mv, it.Close() } // getModuleVersion gets the version for a given module, and returns true if it exists, false otherwise -func (k Keeper) getModuleVersion(ctx sdk.Context, name string) (uint64, bool) { - store := ctx.KVStore(k.storeKey) - it := storetypes.KVStorePrefixIterator(store, []byte{types.VersionMapByte}) - defer it.Close() +func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, bool, error) { + store := k.storeService.OpenKVStore(ctx) + prefix := []byte{types.VersionMapByte} + it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) + if err != nil { + return 0, false, err + } for ; it.Valid(); it.Next() { moduleName := string(it.Key()[1:]) if moduleName == name { version := binary.BigEndian.Uint64(it.Value()) - return version, true + return version, true, nil } } - return 0, false + + return 0, false, it.Close() } // ScheduleUpgrade schedules an upgrade based on the specified plan. // If there is another Plan already scheduled, it will cancel and overwrite it. // ScheduleUpgrade will also write the upgraded IBC ClientState to the upgraded client // path if it is specified in the plan. -func (k Keeper) ScheduleUpgrade(ctx sdk.Context, plan types.Plan) error { +func (k Keeper) ScheduleUpgrade(ctx context.Context, plan types.Plan) error { if err := plan.ValidateBasic(); err != nil { return err } // NOTE: allow for the possibility of chains to schedule upgrades in begin block of the same block // as a strategy for emergency hard fork recoveries - if plan.Height < ctx.BlockHeight() { + sdkCtx := sdk.UnwrapSDKContext(ctx) + if plan.Height < sdkCtx.BlockHeight() { return errorsmod.Wrap(sdkerrors.ErrInvalidRequest, "upgrade cannot be scheduled in the past") } - if k.GetDoneHeight(ctx, plan.Name) != 0 { + doneHeight, err := k.GetDoneHeight(ctx, plan.Name) + if err != nil { + return err + } + + if doneHeight != 0 { return errorsmod.Wrapf(sdkerrors.ErrInvalidRequest, "upgrade with name %s has already been completed", plan.Name) } - store := ctx.KVStore(k.storeKey) + store := k.storeService.OpenKVStore(ctx) // clear any old IBC state stored by previous plan - oldPlan, found := k.GetUpgradePlan(ctx) + oldPlan, found, err := k.GetUpgradePlan(ctx) + if err != nil { + return err + } + if found { - k.ClearIBCState(ctx, oldPlan.Height) + err = k.ClearIBCState(ctx, oldPlan.Height) + if err != nil { + return err + } } - bz := k.cdc.MustMarshal(&plan) - store.Set(types.PlanKey(), bz) + bz, err := k.cdc.Marshal(&plan) + if err != nil { + return err + } + + err = store.Set(types.PlanKey(), bz) + if err != nil { + return err + } telemetry.SetGaugeWithLabels([]string{"server", "info"}, 1, []metrics.Label{telemetry.NewLabel("upgrade_height", strconv.FormatInt(plan.Height, 10))}) @@ -232,52 +276,55 @@ func (k Keeper) ScheduleUpgrade(ctx sdk.Context, plan types.Plan) error { } // SetUpgradedClient sets the expected upgraded client for the next version of this chain at the last height the current chain will commit. -func (k Keeper) SetUpgradedClient(ctx sdk.Context, planHeight int64, bz []byte) error { - store := ctx.KVStore(k.storeKey) - store.Set(types.UpgradedClientKey(planHeight), bz) - return nil +func (k Keeper) SetUpgradedClient(ctx context.Context, planHeight int64, bz []byte) error { + store := k.storeService.OpenKVStore(ctx) + return store.Set(types.UpgradedClientKey(planHeight), bz) } // GetUpgradedClient gets the expected upgraded client for the next version of this chain -func (k Keeper) GetUpgradedClient(ctx sdk.Context, height int64) ([]byte, bool) { - store := ctx.KVStore(k.storeKey) - bz := store.Get(types.UpgradedClientKey(height)) - if len(bz) == 0 { - return nil, false +func (k Keeper) GetUpgradedClient(ctx context.Context, height int64) ([]byte, bool, error) { + store := k.storeService.OpenKVStore(ctx) + bz, err := store.Get(types.UpgradedClientKey(height)) + if len(bz) == 0 || err != nil { + return nil, false, err } - return bz, true + return bz, true, nil } // SetUpgradedConsensusState sets the expected upgraded consensus state for the next version of this chain // using the last height committed on this chain. -func (k Keeper) SetUpgradedConsensusState(ctx sdk.Context, planHeight int64, bz []byte) error { - store := ctx.KVStore(k.storeKey) - store.Set(types.UpgradedConsStateKey(planHeight), bz) - return nil +func (k Keeper) SetUpgradedConsensusState(ctx context.Context, planHeight int64, bz []byte) error { + store := k.storeService.OpenKVStore(ctx) + return store.Set(types.UpgradedConsStateKey(planHeight), bz) } // GetUpgradedConsensusState gets the expected upgraded consensus state for the next version of this chain -func (k Keeper) GetUpgradedConsensusState(ctx sdk.Context, lastHeight int64) ([]byte, bool) { - store := ctx.KVStore(k.storeKey) - bz := store.Get(types.UpgradedConsStateKey(lastHeight)) - if len(bz) == 0 { - return nil, false +func (k Keeper) GetUpgradedConsensusState(ctx context.Context, lastHeight int64) ([]byte, bool, error) { + store := k.storeService.OpenKVStore(ctx) + bz, err := store.Get(types.UpgradedConsStateKey(lastHeight)) + if len(bz) == 0 || err != nil { + return nil, false, err } - return bz, true + return bz, true, nil } // GetLastCompletedUpgrade returns the last applied upgrade name and height. -func (k Keeper) GetLastCompletedUpgrade(ctx sdk.Context) (string, int64) { - iter := storetypes.KVStoreReversePrefixIterator(ctx.KVStore(k.storeKey), []byte{types.DoneByte}) - defer iter.Close() +func (k Keeper) GetLastCompletedUpgrade(ctx context.Context) (string, int64, error) { + store := k.storeService.OpenKVStore(ctx) + prefix := []byte{types.DoneByte} + it, err := store.ReverseIterator(prefix, storetypes.PrefixEndBytes(prefix)) + if err != nil { + return "", 0, err + } - if iter.Valid() { - return parseDoneKey(iter.Key()) + if it.Valid() { + name, height := parseDoneKey(it.Key()) + return name, height, it.Close() } - return "", 0 + return "", 0, it.Close() } // parseDoneKey - split upgrade name and height from the done key @@ -298,61 +345,83 @@ func encodeDoneKey(name string, height int64) []byte { } // GetDoneHeight returns the height at which the given upgrade was executed -func (k Keeper) GetDoneHeight(ctx sdk.Context, name string) int64 { - iter := storetypes.KVStorePrefixIterator(ctx.KVStore(k.storeKey), []byte{types.DoneByte}) - defer iter.Close() +func (k Keeper) GetDoneHeight(ctx context.Context, name string) (int64, error) { + store := k.storeService.OpenKVStore(ctx) + prefix := []byte{types.DoneByte} + it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) + if err != nil { + return 0, err + } - for ; iter.Valid(); iter.Next() { - upgradeName, height := parseDoneKey(iter.Key()) + for ; it.Valid(); it.Next() { + upgradeName, height := parseDoneKey(it.Key()) if upgradeName == name { - return height + return height, it.Close() } } - return 0 + + return 0, it.Close() } // ClearIBCState clears any planned IBC state -func (k Keeper) ClearIBCState(ctx sdk.Context, lastHeight int64) { +func (k Keeper) ClearIBCState(ctx context.Context, lastHeight int64) error { // delete IBC client and consensus state from store if this is IBC plan - store := ctx.KVStore(k.storeKey) - store.Delete(types.UpgradedClientKey(lastHeight)) - store.Delete(types.UpgradedConsStateKey(lastHeight)) + store := k.storeService.OpenKVStore(ctx) + err := store.Delete(types.UpgradedClientKey(lastHeight)) + if err != nil { + return err + } + + return store.Delete(types.UpgradedConsStateKey(lastHeight)) } // ClearUpgradePlan clears any schedule upgrade and associated IBC states. -func (k Keeper) ClearUpgradePlan(ctx sdk.Context) { +func (k Keeper) ClearUpgradePlan(ctx context.Context) error { // clear IBC states everytime upgrade plan is removed - oldPlan, found := k.GetUpgradePlan(ctx) + oldPlan, found, err := k.GetUpgradePlan(ctx) + if err != nil { + return err + } + if found { - k.ClearIBCState(ctx, oldPlan.Height) + err := k.ClearIBCState(ctx, oldPlan.Height) + if err != nil { + return err + } } - store := ctx.KVStore(k.storeKey) - store.Delete(types.PlanKey()) + store := k.storeService.OpenKVStore(ctx) + return store.Delete(types.PlanKey()) } // Logger returns a module-specific logger. -func (k Keeper) Logger(ctx sdk.Context) log.Logger { - return ctx.Logger().With("module", "x/"+types.ModuleName) +func (k Keeper) Logger(ctx context.Context) log.Logger { + sdkCtx := sdk.UnwrapSDKContext(ctx) + return sdkCtx.Logger().With("module", "x/"+types.ModuleName) } // GetUpgradePlan returns the currently scheduled Plan if any, setting havePlan to true if there is a scheduled // upgrade or false if there is none -func (k Keeper) GetUpgradePlan(ctx sdk.Context) (plan types.Plan, havePlan bool) { - store := ctx.KVStore(k.storeKey) - bz := store.Get(types.PlanKey()) - if bz == nil { - return plan, false +func (k Keeper) GetUpgradePlan(ctx context.Context) (plan types.Plan, havePlan bool, err error) { + store := k.storeService.OpenKVStore(ctx) + bz, err := store.Get(types.PlanKey()) + if bz == nil || err != nil { + return plan, false, err + } + + err = k.cdc.Unmarshal(bz, &plan) + if err != nil { + return plan, false, err } - k.cdc.MustUnmarshal(bz, &plan) - return plan, true + return plan, true, err } // setDone marks this upgrade name as being done so the name can't be reused accidentally -func (k Keeper) setDone(ctx sdk.Context, name string) { - store := ctx.KVStore(k.storeKey) - store.Set(encodeDoneKey(name, ctx.BlockHeight()), []byte{1}) +func (k Keeper) setDone(ctx context.Context, name string) error { + store := k.storeService.OpenKVStore(ctx) + sdkCtx := sdk.UnwrapSDKContext(ctx) + return store.Set(encodeDoneKey(name, sdkCtx.BlockHeight()), []byte{1}) } // HasHandler returns true iff there is a handler registered for this name @@ -362,22 +431,38 @@ func (k Keeper) HasHandler(name string) bool { } // ApplyUpgrade will execute the handler associated with the Plan and mark the plan as done. -func (k Keeper) ApplyUpgrade(ctx sdk.Context, plan types.Plan) { +func (k Keeper) ApplyUpgrade(ctx context.Context, plan types.Plan) error { handler := k.upgradeHandlers[plan.Name] if handler == nil { - panic("ApplyUpgrade should never be called without first checking HasHandler") + return fmt.Errorf("ApplyUpgrade should never be called without first checking HasHandler") + } + + vm, err := k.GetModuleVersionMap(ctx) + if err != nil { + return err } - updatedVM, err := handler(ctx, plan, k.GetModuleVersionMap(ctx)) + updatedVM, err := handler(ctx, plan, vm) if err != nil { - panic(err) + return err } - k.SetModuleVersionMap(ctx, updatedVM) + err = k.SetModuleVersionMap(ctx, updatedVM) + if err != nil { + return err + } // incremement the protocol version and set it in state and baseapp - nextProtocolVersion := k.getProtocolVersion(ctx) + 1 - k.setProtocolVersion(ctx, nextProtocolVersion) + nextProtocolVersion, err := k.getProtocolVersion(ctx) + if err != nil { + return err + } + nextProtocolVersion++ + err = k.setProtocolVersion(ctx, nextProtocolVersion) + if err != nil { + return err + } + if k.versionSetter != nil { // set protocol version on BaseApp k.versionSetter.SetProtocolVersion(nextProtocolVersion) @@ -385,9 +470,17 @@ func (k Keeper) ApplyUpgrade(ctx sdk.Context, plan types.Plan) { // Must clear IBC state after upgrade is applied as it is stored separately from the upgrade plan. // This will prevent resubmission of upgrade msg after upgrade is already completed. - k.ClearIBCState(ctx, plan.Height) - k.ClearUpgradePlan(ctx) - k.setDone(ctx, plan.Name) + err = k.ClearIBCState(ctx, plan.Height) + if err != nil { + return err + } + + err = k.ClearUpgradePlan(ctx) + if err != nil { + return err + } + + return k.setDone(ctx, plan.Name) } // IsSkipHeight checks if the given height is part of skipUpgradeHeights diff --git a/x/upgrade/keeper/keeper_test.go b/x/upgrade/keeper/keeper_test.go index 0e699806d18b..2f8f0ecb3639 100644 --- a/x/upgrade/keeper/keeper_test.go +++ b/x/upgrade/keeper/keeper_test.go @@ -1,6 +1,7 @@ package keeper_test import ( + "context" "path/filepath" "testing" "time" @@ -15,6 +16,7 @@ import ( "cosmossdk.io/x/upgrade/types" "github.com/cosmos/cosmos-sdk/baseapp" + "github.com/cosmos/cosmos-sdk/runtime" "github.com/cosmos/cosmos-sdk/testutil" simtestutil "github.com/cosmos/cosmos-sdk/testutil/sims" sdk "github.com/cosmos/cosmos-sdk/types" @@ -40,6 +42,7 @@ type KeeperTestSuite struct { func (s *KeeperTestSuite) SetupTest() { s.encCfg = moduletestutil.MakeTestEncodingConfig(upgrade.AppModuleBasic{}) key := storetypes.NewKVStoreKey(types.StoreKey) + storeService := runtime.NewKVStoreService(key) s.key = key testCtx := testutil.DefaultContextWithDB(s.T(), key, storetypes.NewTransientStoreKey("transient_test")) @@ -53,7 +56,7 @@ func (s *KeeperTestSuite) SetupTest() { skipUpgradeHeights := make(map[int64]bool) homeDir := filepath.Join(s.T().TempDir(), "x_upgrade_keeper_test") - s.upgradeKeeper = keeper.NewKeeper(skipUpgradeHeights, key, s.encCfg.Codec, homeDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + s.upgradeKeeper = keeper.NewKeeper(skipUpgradeHeights, storeService, s.encCfg.Codec, homeDir, nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) s.upgradeKeeper.SetVersionSetter(s.baseApp) vs := s.upgradeKeeper.GetVersionSetter() @@ -146,7 +149,7 @@ func (s *KeeperTestSuite) TestScheduleUpgrade() { Height: 123450000, }, setup: func() { - s.upgradeKeeper.SetUpgradeHandler("all-good", func(ctx sdk.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { + s.upgradeKeeper.SetUpgradeHandler("all-good", func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) s.upgradeKeeper.ApplyUpgrade(s.ctx, types.Plan{ @@ -212,7 +215,9 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { // setup test case tc.setup() - gotCs, exists := s.upgradeKeeper.GetUpgradedClient(s.ctx, tc.height) + gotCs, exists, err := s.upgradeKeeper.GetUpgradedClient(s.ctx, tc.height) + s.Require().NoError(err) + if tc.exists { s.Require().Equal(cs, gotCs, "valid case: %s did not retrieve correct client state", tc.name) s.Require().True(exists, "valid case: %s did not retrieve client state", tc.name) @@ -228,7 +233,8 @@ func (s *KeeperTestSuite) TestIsSkipHeight() { ok := s.upgradeKeeper.IsSkipHeight(11) s.Require().False(ok) skip := map[int64]bool{skipOne: true} - upgradeKeeper := keeper.NewKeeper(skip, s.key, s.encCfg.Codec, s.T().TempDir(), nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + storeService := runtime.NewKVStoreService(s.key) + upgradeKeeper := keeper.NewKeeper(skip, storeService, s.encCfg.Codec, s.T().TempDir(), nil, authtypes.NewModuleAddress(govtypes.ModuleName).String()) upgradeKeeper.SetVersionSetter(s.baseApp) s.Require().True(upgradeKeeper.IsSkipHeight(9)) s.Require().False(upgradeKeeper.IsSkipHeight(10)) @@ -237,9 +243,10 @@ func (s *KeeperTestSuite) TestIsSkipHeight() { func (s *KeeperTestSuite) TestUpgradedConsensusState() { cs := []byte("IBC consensus state") s.Require().NoError(s.upgradeKeeper.SetUpgradedConsensusState(s.ctx, 10, cs)) - bz, ok := s.upgradeKeeper.GetUpgradedConsensusState(s.ctx, 10) + bz, ok, err := s.upgradeKeeper.GetUpgradedConsensusState(s.ctx, 10) s.Require().True(ok) s.Require().Equal(cs, bz) + s.Require().NoError(err) } func (s *KeeperTestSuite) TestDowngradeVerified() { @@ -259,13 +266,11 @@ func (s *KeeperTestSuite) TestIncrementProtocolVersion() { Info: "some text here", Height: 100, } - s.Require().PanicsWithValue("ApplyUpgrade should never be called without first checking HasHandler", - func() { - s.upgradeKeeper.ApplyUpgrade(s.ctx, dummyPlan) - }, - ) - s.upgradeKeeper.SetUpgradeHandler("dummy", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) + err := s.upgradeKeeper.ApplyUpgrade(s.ctx, dummyPlan) + s.Require().EqualError(err, "ApplyUpgrade should never be called without first checking HasHandler") + + s.upgradeKeeper.SetUpgradeHandler("dummy", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) s.upgradeKeeper.ApplyUpgrade(s.ctx, dummyPlan) upgradedProtocolVersion := s.baseApp.AppVersion() @@ -277,8 +282,10 @@ func (s *KeeperTestSuite) TestIncrementProtocolVersion() { func (s *KeeperTestSuite) TestMigrations() { initialVM := module.VersionMap{"bank": uint64(1)} s.upgradeKeeper.SetModuleVersionMap(s.ctx, initialVM) - vmBefore := s.upgradeKeeper.GetModuleVersionMap(s.ctx) - s.upgradeKeeper.SetUpgradeHandler("dummy", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + vmBefore, err := s.upgradeKeeper.GetModuleVersionMap(s.ctx) + s.Require().NoError(err) + + s.upgradeKeeper.SetUpgradeHandler("dummy", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { // simulate upgrading the bank module vm["bank"]++ return vm, nil @@ -290,8 +297,9 @@ func (s *KeeperTestSuite) TestMigrations() { } s.upgradeKeeper.ApplyUpgrade(s.ctx, dummyPlan) - vm := s.upgradeKeeper.GetModuleVersionMap(s.ctx) + vm, err := s.upgradeKeeper.GetModuleVersionMap(s.ctx) s.Require().Equal(vmBefore["bank"]+1, vm["bank"]) + s.Require().NoError(err) } func (s *KeeperTestSuite) TestLastCompletedUpgrade() { @@ -299,11 +307,12 @@ func (s *KeeperTestSuite) TestLastCompletedUpgrade() { require := s.Require() s.T().Log("verify empty name if applied upgrades are empty") - name, height := keeper.GetLastCompletedUpgrade(s.ctx) + name, height, err := keeper.GetLastCompletedUpgrade(s.ctx) require.Equal("", name) require.Equal(int64(0), height) + require.NoError(err) - keeper.SetUpgradeHandler("test0", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + keeper.SetUpgradeHandler("test0", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) @@ -313,11 +322,12 @@ func (s *KeeperTestSuite) TestLastCompletedUpgrade() { }) s.T().Log("verify valid upgrade name and height") - name, height = keeper.GetLastCompletedUpgrade(s.ctx) + name, height, err = keeper.GetLastCompletedUpgrade(s.ctx) require.Equal("test0", name) require.Equal(int64(10), height) + require.NoError(err) - keeper.SetUpgradeHandler("test1", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + keeper.SetUpgradeHandler("test1", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) @@ -328,9 +338,10 @@ func (s *KeeperTestSuite) TestLastCompletedUpgrade() { }) s.T().Log("verify valid upgrade name and height with multiple upgrades") - name, height = keeper.GetLastCompletedUpgrade(newCtx) + name, height, err = keeper.GetLastCompletedUpgrade(newCtx) require.Equal("test1", name) require.Equal(int64(15), height) + require.NoError(err) } // This test ensures that `GetLastDoneUpgrade` always returns the last upgrade according to the block height @@ -340,7 +351,7 @@ func (s *KeeperTestSuite) TestLastCompletedUpgradeOrdering() { require := s.Require() // apply first upgrade - keeper.SetUpgradeHandler("test-v0.9", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + keeper.SetUpgradeHandler("test-v0.9", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) @@ -349,24 +360,27 @@ func (s *KeeperTestSuite) TestLastCompletedUpgradeOrdering() { Height: 10, }) - name, height := keeper.GetLastCompletedUpgrade(s.ctx) + name, height, err := keeper.GetLastCompletedUpgrade(s.ctx) require.Equal("test-v0.9", name) require.Equal(int64(10), height) + require.NoError(err) // apply second upgrade - keeper.SetUpgradeHandler("test-v0.10", func(_ sdk.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { + keeper.SetUpgradeHandler("test-v0.10", func(_ context.Context, _ types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) newCtx := s.ctx.WithBlockHeight(15) - keeper.ApplyUpgrade(newCtx, types.Plan{ + err = keeper.ApplyUpgrade(newCtx, types.Plan{ Name: "test-v0.10", Height: 15, }) + require.NoError(err) - name, height = keeper.GetLastCompletedUpgrade(newCtx) + name, height, err = keeper.GetLastCompletedUpgrade(newCtx) require.Equal("test-v0.10", name) require.Equal(int64(15), height) + require.NoError(err) } func TestKeeperTestSuite(t *testing.T) { diff --git a/x/upgrade/keeper/migrations.go b/x/upgrade/keeper/migrations.go index 206623af2b1a..3f210f57673a 100644 --- a/x/upgrade/keeper/migrations.go +++ b/x/upgrade/keeper/migrations.go @@ -4,9 +4,11 @@ import ( "encoding/binary" "cosmossdk.io/store/prefix" - storetypes "cosmossdk.io/store/types" + // storetypes "cosmossdk.io/store/types" + storetypes "cosmossdk.io/core/store" "cosmossdk.io/x/upgrade/types" + "github.com/cosmos/cosmos-sdk/runtime" sdk "github.com/cosmos/cosmos-sdk/types" ) @@ -22,12 +24,12 @@ func NewMigrator(keeper *Keeper) Migrator { // Migrate1to2 migrates from version 1 to 2. func (m Migrator) Migrate1to2(ctx sdk.Context) error { - return migrateDoneUpgradeKeys(ctx, m.keeper.storeKey) + return migrateDoneUpgradeKeys(ctx, m.keeper.storeService) } -func migrateDoneUpgradeKeys(ctx sdk.Context, storeKey storetypes.StoreKey) error { - store := ctx.KVStore(storeKey) - oldDoneStore := prefix.NewStore(store, []byte{types.DoneByte}) +func migrateDoneUpgradeKeys(ctx sdk.Context, storeService storetypes.KVStoreService) error { + store := storeService.OpenKVStore(ctx) + oldDoneStore := prefix.NewStore(runtime.KVStoreAdapter(store), []byte{types.DoneByte}) oldDoneStoreIter := oldDoneStore.Iterator(nil, nil) defer oldDoneStoreIter.Close() @@ -37,7 +39,11 @@ func migrateDoneUpgradeKeys(ctx sdk.Context, storeKey storetypes.StoreKey) error upgradeHeight := int64(binary.BigEndian.Uint64(oldDoneStoreIter.Value())) newKey := encodeDoneKey(upgradeName, upgradeHeight) - store.Set(newKey, []byte{1}) + err := store.Set(newKey, []byte{1}) + if err != nil { + return err + } + oldDoneStore.Delete(oldKey) } return nil diff --git a/x/upgrade/keeper/migrations_test.go b/x/upgrade/keeper/migrations_test.go index cbcda24f6a45..47225bcef911 100644 --- a/x/upgrade/keeper/migrations_test.go +++ b/x/upgrade/keeper/migrations_test.go @@ -8,6 +8,7 @@ import ( "cosmossdk.io/x/upgrade/types" "github.com/stretchr/testify/require" + "github.com/cosmos/cosmos-sdk/runtime" "github.com/cosmos/cosmos-sdk/testutil" ) @@ -22,8 +23,9 @@ func encodeOldDoneKey(upgrade storedUpgrade) []byte { func TestMigrateDoneUpgradeKeys(t *testing.T) { upgradeKey := storetypes.NewKVStoreKey("upgrade") + storeService := runtime.NewKVStoreService(upgradeKey) ctx := testutil.DefaultContext(upgradeKey, storetypes.NewTransientStoreKey("transient_test")) - store := ctx.KVStore(upgradeKey) + store := storeService.OpenKVStore(ctx) testCases := []struct { name string @@ -44,17 +46,22 @@ func TestMigrateDoneUpgradeKeys(t *testing.T) { bz := make([]byte, 8) binary.BigEndian.PutUint64(bz, uint64(upgrade.height)) oldKey := encodeOldDoneKey(upgrade) - store.Set(oldKey, bz) + require.NoError(t, store.Set(oldKey, bz)) } - err := migrateDoneUpgradeKeys(ctx, upgradeKey) + err := migrateDoneUpgradeKeys(ctx, storeService) require.NoError(t, err) for _, upgrade := range tc.upgrades { newKey := encodeDoneKey(upgrade.name, upgrade.height) oldKey := encodeOldDoneKey(upgrade) - require.Nil(t, store.Get(oldKey)) - require.Equal(t, []byte{1}, store.Get(newKey)) + v, err := store.Get(oldKey) + require.Nil(t, v) + require.NoError(t, err) + + nv, err := store.Get(newKey) + require.Equal(t, []byte{1}, nv) + require.NoError(t, err) } } } diff --git a/x/upgrade/keeper/msg_server_test.go b/x/upgrade/keeper/msg_server_test.go index aba54c5cd8dc..5647b81db873 100644 --- a/x/upgrade/keeper/msg_server_test.go +++ b/x/upgrade/keeper/msg_server_test.go @@ -2,6 +2,7 @@ package keeper_test import ( "cosmossdk.io/x/upgrade/types" + sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/address" ) @@ -73,9 +74,11 @@ func (s *KeeperTestSuite) TestSoftwareUpgrade() { s.Require().Contains(err.Error(), tc.errMsg) } else { s.Require().NoError(err) - plan, found := s.upgradeKeeper.GetUpgradePlan(s.ctx) + plan, found, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) + s.Require().NoError(err) s.Require().Equal(true, found) s.Require().Equal(tc.req.Plan, plan) + } }) } @@ -130,7 +133,8 @@ func (s *KeeperTestSuite) TestCancelUpgrade() { s.Require().Contains(err.Error(), tc.errMsg) } else { s.Require().NoError(err) - _, found := s.upgradeKeeper.GetUpgradePlan(s.ctx) + _, found, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) + s.Require().NoError(err) s.Require().Equal(false, found) } }) diff --git a/x/upgrade/module.go b/x/upgrade/module.go index 77938040fbf3..181cb9b84719 100644 --- a/x/upgrade/module.go +++ b/x/upgrade/module.go @@ -15,7 +15,7 @@ import ( "cosmossdk.io/core/appmodule" "cosmossdk.io/depinject" - store "cosmossdk.io/store/types" + "cosmossdk.io/core/store" "cosmossdk.io/x/upgrade/client/cli" "cosmossdk.io/x/upgrade/keeper" "cosmossdk.io/x/upgrade/types" @@ -124,13 +124,21 @@ func (am AppModule) InitGenesis(ctx sdk.Context, _ codec.JSONCodec, _ json.RawMe if versionMap := am.keeper.GetInitVersionMap(); versionMap != nil { // chains can still use a custom init chainer for setting the version map // this means that we need to combine the manually wired modules version map with app wiring enabled modules version map - for name, version := range am.keeper.GetModuleVersionMap(ctx) { + moduleVM, err := am.keeper.GetModuleVersionMap(ctx) + if err != nil { + panic(err) + } + + for name, version := range moduleVM { if _, ok := versionMap[name]; !ok { versionMap[name] = version } } - am.keeper.SetModuleVersionMap(ctx, versionMap) + err = am.keeper.SetModuleVersionMap(ctx, versionMap) + if err != nil { + panic(err) + } } return []abci.ValidatorUpdate{} @@ -158,9 +166,7 @@ func (AppModule) ConsensusVersion() uint64 { return ConsensusVersion } // // CONTRACT: this is registered in BeginBlocker *before* all other modules' BeginBlock functions func (am AppModule) BeginBlock(ctx context.Context) error { - c := sdk.UnwrapSDKContext(ctx) - BeginBlocker(am.keeper, c) - return nil + return BeginBlocker(am.keeper, ctx) } // @@ -178,7 +184,7 @@ type ModuleInputs struct { depinject.In Config *modulev1.Module - Key *store.KVStoreKey + StoreService store.KVStoreService Cdc codec.Codec AddressCodec address.Codec @@ -215,7 +221,7 @@ func ProvideModule(in ModuleInputs) ModuleOutputs { } // set the governance module account as the authority for conducting upgrades - k := keeper.NewKeeper(skipUpgradeHeights, in.Key, in.Cdc, homePath, nil, authority.String()) + k := keeper.NewKeeper(skipUpgradeHeights, in.StoreService, in.Cdc, homePath, nil, authority.String()) baseappOpt := func(app *baseapp.BaseApp) { k.SetVersionSetter(app) } diff --git a/x/upgrade/types/handler.go b/x/upgrade/types/handler.go index 0543b6bbeb2d..80d57da469e2 100644 --- a/x/upgrade/types/handler.go +++ b/x/upgrade/types/handler.go @@ -1,7 +1,8 @@ package types import ( - sdk "github.com/cosmos/cosmos-sdk/types" + context "context" + "github.com/cosmos/cosmos-sdk/types/module" ) @@ -23,4 +24,4 @@ import ( // function. // // Please also refer to docs/core/upgrade.md for more information. -type UpgradeHandler func(ctx sdk.Context, plan Plan, fromVM module.VersionMap) (module.VersionMap, error) +type UpgradeHandler func(ctx context.Context, plan Plan, fromVM module.VersionMap) (module.VersionMap, error) diff --git a/x/upgrade/types/plan.go b/x/upgrade/types/plan.go index 7f0659ba4eaf..8b92fe006031 100644 --- a/x/upgrade/types/plan.go +++ b/x/upgrade/types/plan.go @@ -5,7 +5,6 @@ import ( errorsmod "cosmossdk.io/errors" - sdk "github.com/cosmos/cosmos-sdk/types" sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" ) @@ -30,12 +29,9 @@ func (p Plan) ValidateBasic() error { return nil } -// ShouldExecute returns true if the Plan is ready to execute given the current context -func (p Plan) ShouldExecute(ctx sdk.Context) bool { - if p.Height > 0 { - return p.Height <= ctx.BlockHeight() - } - return false +// ShouldExecute returns true if the Plan is ready to execute given the current block height +func (p Plan) ShouldExecute(blockHeight int64) bool { + return p.Height > 0 && p.Height <= blockHeight } // DueAt is a string representation of when this plan is due to be executed diff --git a/x/upgrade/types/plan_test.go b/x/upgrade/types/plan_test.go index 4ef19b1ff0d2..423f12b1d47f 100644 --- a/x/upgrade/types/plan_test.go +++ b/x/upgrade/types/plan_test.go @@ -4,15 +4,12 @@ import ( "testing" "time" - cmtproto "github.com/cometbft/cometbft/proto/tendermint/types" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" - "cosmossdk.io/log" "cosmossdk.io/x/upgrade/types" codectypes "github.com/cosmos/cosmos-sdk/codec/types" - sdk "github.com/cosmos/cosmos-sdk/types" ) func mustParseTime(s string) time.Time { @@ -147,8 +144,7 @@ func TestShouldExecute(t *testing.T) { for name, tc := range cases { tc := tc // copy to local variable for scopelint t.Run(name, func(t *testing.T) { - ctx := sdk.NewContext(nil, cmtproto.Header{Height: tc.ctxHeight, Time: tc.ctxTime}, false, log.NewNopLogger()) - should := tc.p.ShouldExecute(ctx) + should := tc.p.ShouldExecute(tc.ctxHeight) assert.Equal(t, tc.expected, should) }) } From 5ffc5362e3b2d5f70308b51526f759c78314b41f Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 10:43:18 +0200 Subject: [PATCH 03/24] progress --- simapp/app.go | 3 ++- simapp/upgrades.go | 5 +++-- types/module/configurator.go | 9 ++++++--- types/module/module.go | 11 ++++++----- 4 files changed, 17 insertions(+), 11 deletions(-) diff --git a/simapp/app.go b/simapp/app.go index 195d2d0dac11..85f2c8c06bb2 100644 --- a/simapp/app.go +++ b/simapp/app.go @@ -40,6 +40,7 @@ import ( "cosmossdk.io/x/circuit" circuitkeeper "cosmossdk.io/x/circuit/keeper" circuittypes "cosmossdk.io/x/circuit/types" + "github.com/cosmos/cosmos-sdk/baseapp" "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/flags" @@ -318,7 +319,7 @@ func NewSimApp( } homePath := cast.ToString(appOpts.Get(flags.FlagHome)) // set the governance module account as the authority for conducting upgrades - app.UpgradeKeeper = upgradekeeper.NewKeeper(skipUpgradeHeights, keys[upgradetypes.StoreKey], appCodec, homePath, app.BaseApp, authtypes.NewModuleAddress(govtypes.ModuleName).String()) + app.UpgradeKeeper = upgradekeeper.NewKeeper(skipUpgradeHeights, runtime.NewKVStoreService(keys[upgradetypes.StoreKey]), appCodec, homePath, app.BaseApp, authtypes.NewModuleAddress(govtypes.ModuleName).String()) // Register the proposal types // Deprecated: Avoid adding new handlers, instead use the new proposal flow diff --git a/simapp/upgrades.go b/simapp/upgrades.go index 1bce6df3f6ac..6ddb3db02085 100644 --- a/simapp/upgrades.go +++ b/simapp/upgrades.go @@ -1,10 +1,11 @@ package simapp import ( + "context" + storetypes "cosmossdk.io/store/types" upgradetypes "cosmossdk.io/x/upgrade/types" - sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/module" ) @@ -19,7 +20,7 @@ const UpgradeName = "v047-to-v048" func (app SimApp) RegisterUpgradeHandlers() { app.UpgradeKeeper.SetUpgradeHandler( UpgradeName, - func(ctx sdk.Context, _ upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { + func(ctx context.Context, _ upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { return app.ModuleManager.RunMigrations(ctx, app.Configurator(), fromVM) }, ) diff --git a/types/module/configurator.go b/types/module/configurator.go index 58c9747442d4..199b5b8a2290 100644 --- a/types/module/configurator.go +++ b/types/module/configurator.go @@ -1,6 +1,7 @@ package module import ( + "context" "fmt" cosmosmsg "cosmossdk.io/api/cosmos/msg/v1" @@ -127,7 +128,7 @@ func (c *configurator) RegisterMigration(moduleName string, fromVersion uint64, // runModuleMigrations runs all in-place store migrations for one given module from a // version to another version. -func (c *configurator) runModuleMigrations(ctx sdk.Context, moduleName string, fromVersion, toVersion uint64) error { +func (c *configurator) runModuleMigrations(ctx context.Context, moduleName string, fromVersion, toVersion uint64) error { // No-op if toVersion is the initial version or if the version is unchanged. if toVersion <= 1 || fromVersion == toVersion { return nil @@ -144,9 +145,11 @@ func (c *configurator) runModuleMigrations(ctx sdk.Context, moduleName string, f if !found { return errorsmod.Wrapf(sdkerrors.ErrNotFound, "no migration found for module %s from version %d to version %d", moduleName, i, i+1) } - ctx.Logger().Info(fmt.Sprintf("migrating module %s from version %d to version %d", moduleName, i, i+1)) - err := migrateFn(ctx) + sdkCtx := sdk.UnwrapSDKContext(ctx) + sdkCtx.Logger().Info(fmt.Sprintf("migrating module %s from version %d to version %d", moduleName, i, i+1)) + + err := migrateFn(sdkCtx) if err != nil { return err } diff --git a/types/module/module.go b/types/module/module.go index 673e4965a550..226131812873 100644 --- a/types/module/module.go +++ b/types/module/module.go @@ -609,7 +609,7 @@ type VersionMap map[string]uint64 // Example: // // cfg := module.NewConfigurator(...) -// app.UpgradeKeeper.SetUpgradeHandler("my-plan", func(ctx sdk.Context, plan upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { +// app.UpgradeKeeper.SetUpgradeHandler("my-plan", func(ctx context.Context, plan upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { // return app.mm.RunMigrations(ctx, cfg, fromVM) // }) // @@ -636,7 +636,7 @@ type VersionMap map[string]uint64 // Example: // // cfg := module.NewConfigurator(...) -// app.UpgradeKeeper.SetUpgradeHandler("my-plan", func(ctx sdk.Context, plan upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { +// app.UpgradeKeeper.SetUpgradeHandler("my-plan", func(ctx context.Context, plan upgradetypes.Plan, fromVM module.VersionMap) (module.VersionMap, error) { // // Assume "foo" is a new module. // // `fromVM` is fetched from existing x/upgrade store. Since foo didn't exist // // before this upgrade, `v, exists := fromVM["foo"]; exists == false`, and RunMigration will by default @@ -649,7 +649,7 @@ type VersionMap map[string]uint64 // }) // // Please also refer to docs/core/upgrade.md for more information. -func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { +func (m Manager) RunMigrations(ctx context.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { c, ok := cfg.(*configurator) if !ok { return nil, errorsmod.Wrapf(sdkerrors.ErrInvalidType, "expected %T, got %T", &configurator{}, cfg) @@ -682,9 +682,10 @@ func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM Version return nil, err } } else { - ctx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) + sdkCtx := sdk.UnwrapSDKContext(ctx) + sdkCtx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) if module, ok := m.Modules[moduleName].(HasGenesis); ok { - moduleValUpdates := module.InitGenesis(ctx, c.cdc, module.DefaultGenesis(c.cdc)) + moduleValUpdates := module.InitGenesis(sdkCtx, c.cdc, module.DefaultGenesis(c.cdc)) // The module manager assumes only one module will update the // validator set, and it can't be a new module. if len(moduleValUpdates) > 0 { From 431efcdf2a41ab86966fd02e265e2df634f18a1b Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 10:50:45 +0200 Subject: [PATCH 04/24] fix test --- simapp/app_test.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/simapp/app_test.go b/simapp/app_test.go index 28268ab416dc..858dc99cf95e 100644 --- a/simapp/app_test.go +++ b/simapp/app_test.go @@ -263,7 +263,8 @@ func TestUpgradeStateOnGenesis(t *testing.T) { // make sure the upgrade keeper has version map in state ctx := app.NewContext(false, cmtproto.Header{}) - vm := app.UpgradeKeeper.GetModuleVersionMap(ctx) + vm, err := app.UpgradeKeeper.GetModuleVersionMap(ctx) + require.NoError(t, err) for v, i := range app.ModuleManager.Modules { if i, ok := i.(module.HasConsensusVersion); ok { require.Equal(t, vm[v], i.ConsensusVersion()) From 6688b9790c80b86f30a73217d688deca2ce7acd9 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 11:11:30 +0200 Subject: [PATCH 05/24] progress --- x/upgrade/abci.go | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/x/upgrade/abci.go b/x/upgrade/abci.go index b0911e24e07f..0c25f0b92709 100644 --- a/x/upgrade/abci.go +++ b/x/upgrade/abci.go @@ -86,15 +86,16 @@ func BeginBlocker(k *keeper.Keeper, ctx context.Context) error { upgradeMsg := BuildUpgradeNeededMsg(plan) logger.Error(upgradeMsg) - // TODO: figure out if I can convert this message into an error (will it panic the same way?) - panic(upgradeMsg) + + // Returning an error will end up in a panic + return fmt.Errorf(upgradeMsg) } // We have an upgrade handler for this upgrade name, so apply the upgrade logger.Info(fmt.Sprintf("applying upgrade \"%s\" at %s", plan.Name, plan.DueAt())) sdkCtx = sdkCtx.WithBlockGasMeter(storetypes.NewInfiniteGasMeter()) // TODO: figure out how to pass a block gas meter to the ApplyUpgradeFunction - return k.ApplyUpgrade(sdk.WrapSDKContext(sdkCtx), plan) + return k.ApplyUpgrade(sdkCtx, plan) } // if we have a pending upgrade, but it is not yet time, make sure we did not @@ -102,8 +103,9 @@ func BeginBlocker(k *keeper.Keeper, ctx context.Context) error { if k.HasHandler(plan.Name) { downgradeMsg := fmt.Sprintf("BINARY UPDATED BEFORE TRIGGER! UPGRADE \"%s\" - in binary but not executed on chain. Downgrade your binary", plan.Name) logger.Error(downgradeMsg) - // TODO: figure out if I can convert this message into an error (will it panic the same way?) - panic(downgradeMsg) + + // Returning an error will end up in a panic + return fmt.Errorf(downgradeMsg) } return nil From d0ff944aede2d386a539f65ec263a797aad5faf4 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 11:24:20 +0200 Subject: [PATCH 06/24] fix test --- x/upgrade/abci_test.go | 61 +++++++++++++++++++----------------------- 1 file changed, 28 insertions(+), 33 deletions(-) diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index be502360c47c..ffa538fa8dd8 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -106,34 +106,33 @@ func VerifyDoUpgrade(t *testing.T) { t.Log("Verify that a panic happens at the upgrade height") newCtx := s.ctx.WithBlockHeight(s.ctx.BlockHeight() + 1).WithBlockTime(time.Now()) - require.Panics(t, func() { - s.module.BeginBlock(newCtx) - }) + err := s.module.BeginBlock(newCtx) + require.ErrorContains(t, err, "UPGRADE \"test\" NEEDED at height: 11: ") t.Log("Verify that the upgrade can be successfully applied with a handler") s.keeper.SetUpgradeHandler("test", func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) VerifyCleared(t, newCtx) } func VerifyDoUpgradeWithCtx(t *testing.T, newCtx sdk.Context, proposalName string) { t.Log("Verify that a panic happens at the upgrade height") - require.Panics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err := s.module.BeginBlock(newCtx) + require.ErrorContains(t, err, "UPGRADE \""+proposalName+"\" NEEDED at height: ") t.Log("Verify that the upgrade can be successfully applied with a handler") s.keeper.SetUpgradeHandler(proposalName, func(ctx context.Context, plan types.Plan, vm module.VersionMap) (module.VersionMap, error) { return vm, nil }) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) VerifyCleared(t, newCtx) } @@ -148,25 +147,24 @@ func TestHaltIfTooNew(t *testing.T) { }) newCtx := s.ctx.WithBlockHeight(s.ctx.BlockHeight() + 1).WithBlockTime(time.Now()) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err := s.module.BeginBlock(newCtx) + require.NoError(t, err) require.Equal(t, 0, called) t.Log("Verify we panic if we have a registered handler ahead of time") - err := s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop", Plan: types.Plan{Name: "future", Height: s.ctx.BlockHeight() + 3}}) //nolint:staticcheck // we're testing deprecated code + err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop", Plan: types.Plan{Name: "future", Height: s.ctx.BlockHeight() + 3}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err) - require.Panics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err = s.module.BeginBlock(newCtx) + require.EqualError(t, err, "BINARY UPDATED BEFORE TRIGGER! UPGRADE \"future\" - in binary but not executed on chain. Downgrade your binary") require.Equal(t, 0, called) t.Log("Verify we no longer panic if the plan is on time") futCtx := s.ctx.WithBlockHeight(s.ctx.BlockHeight() + 3).WithBlockTime(time.Now()) - require.NotPanics(t, func() { - s.module.BeginBlock(futCtx) - }) + err = s.module.BeginBlock(futCtx) + require.NoError(t, err) require.Equal(t, 1, called) VerifyCleared(t, futCtx) @@ -369,9 +367,9 @@ func TestUpgradeWithoutSkip(t *testing.T) { err := s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop", Plan: types.Plan{Name: "test", Height: s.ctx.BlockHeight() + 1}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err) t.Log("Verify if upgrade happens without skip upgrade") - require.Panics(t, func() { - s.module.BeginBlock(newCtx) - }) + + err = s.module.BeginBlock(newCtx) + require.ErrorContains(t, err, "UPGRADE \"test\" NEEDED at height:") VerifyDoUpgrade(t) VerifyDone(t, s.ctx, "test") @@ -416,7 +414,7 @@ func TestBinaryVersion(t *testing.T) { testCases := []struct { name string preRun func() sdk.Context - expectPanic bool + expectError bool }{ { "test not panic: no scheduled upgrade or applied upgrade is present", @@ -460,14 +458,11 @@ func TestBinaryVersion(t *testing.T) { for _, tc := range testCases { ctx := tc.preRun() - if tc.expectPanic { - require.Panics(t, func() { - s.module.BeginBlock(ctx) - }) + err := s.module.BeginBlock(ctx) + if tc.expectError { + require.Error(t, err) } else { - require.NotPanics(t, func() { - s.module.BeginBlock(ctx) - }) + require.NoError(t, err) } } } From 0800c56e82a1c197f0954f78f34de4488f7a982d Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 11:30:04 +0200 Subject: [PATCH 07/24] fix test --- x/upgrade/abci_test.go | 35 ++++++++++++++--------------------- 1 file changed, 14 insertions(+), 21 deletions(-) diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index ffa538fa8dd8..71ec786345e3 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -205,9 +205,8 @@ func TestCantApplySameUpgradeTwice(t *testing.T) { func TestNoSpuriousUpgrades(t *testing.T) { s := setupTest(t, 10, map[int64]bool{}) t.Log("Verify that no upgrade panic is triggered in the BeginBlocker when we haven't scheduled an upgrade") - require.NotPanics(t, func() { - s.module.BeginBlock(s.ctx) - }) + err := s.module.BeginBlock(s.ctx) + require.NoError(t, err) } func TestPlanStringer(t *testing.T) { @@ -265,18 +264,16 @@ func TestSkipUpgradeSkippingAll(t *testing.T) { VerifySet(t, map[int64]bool{skipOne: true, skipTwo: true}) newCtx = newCtx.WithBlockHeight(skipOne) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) t.Log("Verify a second proposal also is being cleared") err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop2", Plan: types.Plan{Name: "test2", Height: skipTwo}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err) newCtx = newCtx.WithBlockHeight(skipTwo) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) // To ensure verification is being done only after both upgrades are cleared t.Log("Verify if both proposals are cleared") @@ -302,9 +299,8 @@ func TestUpgradeSkippingOne(t *testing.T) { // Setting block height of proposal test newCtx = newCtx.WithBlockHeight(skipOne) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) t.Log("Verify the second proposal is not skipped") err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop2", Plan: types.Plan{Name: "test2", Height: skipTwo}}) //nolint:staticcheck // we're testing deprecated code @@ -336,18 +332,16 @@ func TestUpgradeSkippingOnlyTwo(t *testing.T) { // Setting block height of proposal test newCtx = newCtx.WithBlockHeight(skipOne) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) // A new proposal with height in skipUpgradeHeights err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop2", Plan: types.Plan{Name: "test2", Height: skipTwo}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err) // Setting block height of proposal test2 newCtx = newCtx.WithBlockHeight(skipTwo) - require.NotPanics(t, func() { - s.module.BeginBlock(newCtx) - }) + err = s.module.BeginBlock(newCtx) + require.NoError(t, err) t.Log("Verify a new proposal is not skipped") err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop3", Plan: types.Plan{Name: "test3", Height: skipThree}}) //nolint:staticcheck // we're testing deprecated code @@ -494,9 +488,8 @@ func TestDowngradeVerification(t *testing.T) { }) // successful upgrade. - require.NotPanics(t, func() { - m.BeginBlock(ctx) - }) + err = s.module.BeginBlock(ctx) + require.NoError(t, err) ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 1) testCases := map[string]struct { From 06f1f74a3101a748c054045ee585e9cad43be66f Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 16:30:33 +0200 Subject: [PATCH 08/24] fix test --- x/upgrade/abci_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index 71ec786345e3..8f97946742c8 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -488,7 +488,7 @@ func TestDowngradeVerification(t *testing.T) { }) // successful upgrade. - err = s.module.BeginBlock(ctx) + err = m.BeginBlock(ctx) require.NoError(t, err) ctx = ctx.WithBlockHeight(ctx.BlockHeight() + 1) From d8f7f4113af49a859c340ff5fc6d0083fe377c97 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 16:36:23 +0200 Subject: [PATCH 09/24] more fixes --- tests/e2e/upgrade/suite.go | 7 +++++-- x/upgrade/abci.go | 4 ++-- x/upgrade/module.go | 2 +- 3 files changed, 8 insertions(+), 5 deletions(-) diff --git a/tests/e2e/upgrade/suite.go b/tests/e2e/upgrade/suite.go index 8e500050b23f..620071329093 100644 --- a/tests/e2e/upgrade/suite.go +++ b/tests/e2e/upgrade/suite.go @@ -76,10 +76,13 @@ func (s *E2ETestSuite) TestModuleVersionsCLI() { // avoid printing as yaml from CLI command clientCtx.OutputFormat = "JSON" - vm := s.upgradeKeeper.GetModuleVersionMap(s.ctx) - mv := s.upgradeKeeper.GetModuleVersions(s.ctx) + vm, err := s.upgradeKeeper.GetModuleVersionMap(s.ctx) + s.Require().NoError(err) s.Require().NotEmpty(vm) + mv, err := s.upgradeKeeper.GetModuleVersions(s.ctx) + s.Require().NoError(err) + for _, tc := range testCases { s.Run(fmt.Sprintf("Case %s", tc.msg), func() { expect := mv diff --git a/x/upgrade/abci.go b/x/upgrade/abci.go index 0c25f0b92709..7d80141163c7 100644 --- a/x/upgrade/abci.go +++ b/x/upgrade/abci.go @@ -22,7 +22,7 @@ import ( // The purpose is to ensure the binary is switched EXACTLY at the desired block, and to allow // a migration to be executed if needed upon this switch (migration defined in the new binary) // skipUpgradeHeightArray is a set of block heights for which the upgrade must be skipped -func BeginBlocker(k *keeper.Keeper, ctx context.Context) error { +func BeginBlocker(ctx context.Context, k *keeper.Keeper) error { defer telemetry.ModuleMeasureSince(types.ModuleName, time.Now(), telemetry.MetricKeyBeginBlocker) sdkCtx := sdk.UnwrapSDKContext(ctx) @@ -53,7 +53,7 @@ func BeginBlocker(k *keeper.Keeper, ctx context.Context) error { appVersion = cp.Version.App } - return fmt.Errorf("Wrong app version %d, upgrade handler is missing for %s upgrade plan", appVersion, lastAppliedPlan) + return fmt.Errorf("wrong app version %d, upgrade handler is missing for %s upgrade plan", appVersion, lastAppliedPlan) } } } diff --git a/x/upgrade/module.go b/x/upgrade/module.go index 181cb9b84719..fa9c51aa291e 100644 --- a/x/upgrade/module.go +++ b/x/upgrade/module.go @@ -166,7 +166,7 @@ func (AppModule) ConsensusVersion() uint64 { return ConsensusVersion } // // CONTRACT: this is registered in BeginBlocker *before* all other modules' BeginBlock functions func (am AppModule) BeginBlock(ctx context.Context) error { - return BeginBlocker(am.keeper, ctx) + return BeginBlocker(ctx, am.keeper) } // From 7779b318c3b3d2d27dd636859e32cd3b7c15b360 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:31:06 +0200 Subject: [PATCH 10/24] undo some changes --- types/module/module.go | 7 +++---- x/upgrade/CHANGELOG.md | 4 ++++ 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/types/module/module.go b/types/module/module.go index 226131812873..517fc363ab1a 100644 --- a/types/module/module.go +++ b/types/module/module.go @@ -649,7 +649,7 @@ type VersionMap map[string]uint64 // }) // // Please also refer to docs/core/upgrade.md for more information. -func (m Manager) RunMigrations(ctx context.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { +func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { c, ok := cfg.(*configurator) if !ok { return nil, errorsmod.Wrapf(sdkerrors.ErrInvalidType, "expected %T, got %T", &configurator{}, cfg) @@ -682,10 +682,9 @@ func (m Manager) RunMigrations(ctx context.Context, cfg Configurator, fromVM Ver return nil, err } } else { - sdkCtx := sdk.UnwrapSDKContext(ctx) - sdkCtx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) + ctx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) if module, ok := m.Modules[moduleName].(HasGenesis); ok { - moduleValUpdates := module.InitGenesis(sdkCtx, c.cdc, module.DefaultGenesis(c.cdc)) + moduleValUpdates := module.InitGenesis(ctx, c.cdc, module.DefaultGenesis(c.cdc)) // The module manager assumes only one module will update the // validator set, and it can't be a new module. if len(moduleValUpdates) > 0 { diff --git a/x/upgrade/CHANGELOG.md b/x/upgrade/CHANGELOG.md index f3968a0a6a4f..a54500441eb4 100644 --- a/x/upgrade/CHANGELOG.md +++ b/x/upgrade/CHANGELOG.md @@ -29,3 +29,7 @@ Ref: https://keepachangelog.com/en/1.0.0/ * [#14880](https://github.com/cosmos/cosmos-sdk/pull/14880) Switch from using gov v1beta1 to gov v1 in upgrade CLIs. * [#14764](https://github.com/cosmos/cosmos-sdk/pull/14764) The `x/upgrade` module is extracted to have a separate go.mod file which allows it be a standalone module. + +### API Breaking + +* (x/upgrade) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error` (instead of panicking or returning a `found bool`). Iterators callback functions now return an error instead of a `bool`. From 0d5a079883ab1e4e9d385c1f53a411746375dc87 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:32:41 +0200 Subject: [PATCH 11/24] undo some changes --- types/module/configurator.go | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/types/module/configurator.go b/types/module/configurator.go index 199b5b8a2290..48df53466eb9 100644 --- a/types/module/configurator.go +++ b/types/module/configurator.go @@ -1,7 +1,6 @@ package module import ( - "context" "fmt" cosmosmsg "cosmossdk.io/api/cosmos/msg/v1" @@ -128,7 +127,7 @@ func (c *configurator) RegisterMigration(moduleName string, fromVersion uint64, // runModuleMigrations runs all in-place store migrations for one given module from a // version to another version. -func (c *configurator) runModuleMigrations(ctx context.Context, moduleName string, fromVersion, toVersion uint64) error { +func (c *configurator) runModuleMigrations(ctx sdk.Context, moduleName string, fromVersion, toVersion uint64) error { // No-op if toVersion is the initial version or if the version is unchanged. if toVersion <= 1 || fromVersion == toVersion { return nil @@ -146,10 +145,9 @@ func (c *configurator) runModuleMigrations(ctx context.Context, moduleName strin return errorsmod.Wrapf(sdkerrors.ErrNotFound, "no migration found for module %s from version %d to version %d", moduleName, i, i+1) } - sdkCtx := sdk.UnwrapSDKContext(ctx) - sdkCtx.Logger().Info(fmt.Sprintf("migrating module %s from version %d to version %d", moduleName, i, i+1)) + ctx.Logger().Info(fmt.Sprintf("migrating module %s from version %d to version %d", moduleName, i, i+1)) - err := migrateFn(sdkCtx) + err := migrateFn(ctx) if err != nil { return err } From e5bbe9f35659ea168333a4dc23b2889e40951ab1 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:33:15 +0200 Subject: [PATCH 12/24] undo some changes --- types/module/configurator.go | 1 - 1 file changed, 1 deletion(-) diff --git a/types/module/configurator.go b/types/module/configurator.go index 48df53466eb9..58c9747442d4 100644 --- a/types/module/configurator.go +++ b/types/module/configurator.go @@ -144,7 +144,6 @@ func (c *configurator) runModuleMigrations(ctx sdk.Context, moduleName string, f if !found { return errorsmod.Wrapf(sdkerrors.ErrNotFound, "no migration found for module %s from version %d to version %d", moduleName, i, i+1) } - ctx.Logger().Info(fmt.Sprintf("migrating module %s from version %d to version %d", moduleName, i, i+1)) err := migrateFn(ctx) From a2939bab2ba533f5122dc5108326d15301abe59f Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:38:46 +0200 Subject: [PATCH 13/24] cl and upgrading --- UPGRADING.md | 2 ++ x/upgrade/CHANGELOG.md | 2 +- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/UPGRADING.md b/UPGRADING.md index 9985c196e936..7145752420ae 100644 --- a/UPGRADING.md +++ b/UPGRADING.md @@ -82,6 +82,7 @@ The following modules `NewKeeper` function now take a `KVStoreService` instead o * `x/feegrant` * `x/gov` * `x/nft` +* `x/upgrade` User manually wiring their chain need to use the `runtime.NewKVStoreService` method to create a `KVStoreService` from a `StoreKey`: @@ -101,6 +102,7 @@ The following modules' `Keeper` methods now take in a `context.Context` instead * `x/distribution` * `x/evidence` * `x/gov` +* `x/upgrade` **Users using depinject do not need any changes, this is automatically done for them.** diff --git a/x/upgrade/CHANGELOG.md b/x/upgrade/CHANGELOG.md index a54500441eb4..17fe463ed82c 100644 --- a/x/upgrade/CHANGELOG.md +++ b/x/upgrade/CHANGELOG.md @@ -32,4 +32,4 @@ Ref: https://keepachangelog.com/en/1.0.0/ ### API Breaking -* (x/upgrade) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error` (instead of panicking or returning a `found bool`). Iterators callback functions now return an error instead of a `bool`. +* (x/upgrade) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error`. Also `UpgradeHandler` now receives a `context.Context`. From f1a924eff059adb3a8321923f6982243f9592a11 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:52:45 +0200 Subject: [PATCH 14/24] RunMigrations now receive context.Context --- types/module/module.go | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/types/module/module.go b/types/module/module.go index 517fc363ab1a..b027b2434280 100644 --- a/types/module/module.go +++ b/types/module/module.go @@ -649,7 +649,7 @@ type VersionMap map[string]uint64 // }) // // Please also refer to docs/core/upgrade.md for more information. -func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { +func (m Manager) RunMigrations(ctx context.Context, cfg Configurator, fromVM VersionMap) (VersionMap, error) { c, ok := cfg.(*configurator) if !ok { return nil, errorsmod.Wrapf(sdkerrors.ErrInvalidType, "expected %T, got %T", &configurator{}, cfg) @@ -659,6 +659,7 @@ func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM Version modules = DefaultMigrationsOrder(m.ModuleNames()) } + sdkCtx := sdk.UnwrapSDKContext(ctx) updatedVM := VersionMap{} for _, moduleName := range modules { module := m.Modules[moduleName] @@ -677,14 +678,14 @@ func (m Manager) RunMigrations(ctx sdk.Context, cfg Configurator, fromVM Version // 2. An existing chain is upgrading from version < 0.43 to v0.43+ for the first time. // In this case, all modules have yet to be added to x/upgrade's VersionMap store. if exists { - err := c.runModuleMigrations(ctx, moduleName, fromVersion, toVersion) + err := c.runModuleMigrations(sdkCtx, moduleName, fromVersion, toVersion) if err != nil { return nil, err } } else { - ctx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) + sdkCtx.Logger().Info(fmt.Sprintf("adding a new module: %s", moduleName)) if module, ok := m.Modules[moduleName].(HasGenesis); ok { - moduleValUpdates := module.InitGenesis(ctx, c.cdc, module.DefaultGenesis(c.cdc)) + moduleValUpdates := module.InitGenesis(sdkCtx, c.cdc, module.DefaultGenesis(c.cdc)) // The module manager assumes only one module will update the // validator set, and it can't be a new module. if len(moduleValUpdates) > 0 { From dfe660a1f107494d550c64613889a4ef9c86f351 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Fri, 19 May 2023 17:56:29 +0200 Subject: [PATCH 15/24] cl++ --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index c3609ae85c08..2b7be3ec4fdb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -129,6 +129,7 @@ Ref: https://keepachangelog.com/en/1.0.0/ ### API Breaking Changes +* (module) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `manager.RunMigrations()` now take a `context.Context` instead of a `sdk.Context`. * (x/gov) [#15988](https://github.com/cosmos/cosmos-sdk/issues/15988) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error` (instead of panicking or returning a `found bool`). Iterators callback functions now return an error instead of a `bool`. * (x/auth) [#15985](https://github.com/cosmos/cosmos-sdk/pull/15985) The `AccountKeeper` does not expose the `QueryServer` and `MsgServer` APIs anymore. * (x/authz) [#15962](https://github.com/cosmos/cosmos-sdk/issues/15962) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context`. The `Authorization` interface's `Accept` method now takes a `context.Context` instead of a `sdk.Context`. From a7665601fcea2eccce8443f844f163b4807b231a Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 10:29:42 +0200 Subject: [PATCH 16/24] handle not founds with errors --- x/upgrade/abci.go | 10 ++-- x/upgrade/abci_test.go | 11 ++-- x/upgrade/handler.go | 3 +- x/upgrade/keeper/grpc_query.go | 37 ++++++------- x/upgrade/keeper/keeper.go | 83 ++++++++++++++++++----------- x/upgrade/keeper/keeper_test.go | 22 ++++---- x/upgrade/keeper/msg_server.go | 5 +- x/upgrade/keeper/msg_server_test.go | 9 ++-- 8 files changed, 97 insertions(+), 83 deletions(-) diff --git a/x/upgrade/abci.go b/x/upgrade/abci.go index 7d80141163c7..ef8197ffa364 100644 --- a/x/upgrade/abci.go +++ b/x/upgrade/abci.go @@ -2,6 +2,7 @@ package upgrade import ( "context" + "errors" "fmt" "time" @@ -9,9 +10,8 @@ import ( "cosmossdk.io/x/upgrade/keeper" "cosmossdk.io/x/upgrade/types" - sdk "github.com/cosmos/cosmos-sdk/types" - "github.com/cosmos/cosmos-sdk/telemetry" + sdk "github.com/cosmos/cosmos-sdk/types" ) // BeginBlock will check if there is a scheduled plan and if it is ready to be executed. @@ -27,10 +27,11 @@ func BeginBlocker(ctx context.Context, k *keeper.Keeper) error { sdkCtx := sdk.UnwrapSDKContext(ctx) blockHeight := sdkCtx.BlockHeight() - plan, found, err := k.GetUpgradePlan(ctx) - if err != nil { + plan, err := k.GetUpgradePlan(ctx) + if err != nil && !errors.Is(err, types.ErrNoUpgradePlanFound) { return err } + found := err == nil if !k.DowngradeVerified() { k.SetDowngradeVerified(true) @@ -94,7 +95,6 @@ func BeginBlocker(ctx context.Context, k *keeper.Keeper) error { // We have an upgrade handler for this upgrade name, so apply the upgrade logger.Info(fmt.Sprintf("applying upgrade \"%s\" at %s", plan.Name, plan.DueAt())) sdkCtx = sdkCtx.WithBlockGasMeter(storetypes.NewInfiniteGasMeter()) - // TODO: figure out how to pass a block gas meter to the ApplyUpgradeFunction return k.ApplyUpgrade(sdkCtx, plan) } diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index 8f97946742c8..66f518c0f8bf 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -172,10 +172,8 @@ func TestHaltIfTooNew(t *testing.T) { func VerifyCleared(t *testing.T, newCtx sdk.Context) { t.Log("Verify that the upgrade plan has been cleared") - plan, _, err := s.keeper.GetUpgradePlan(newCtx) - expected := types.Plan{} - require.Equal(t, plan, expected) - require.NoError(t, err) + _, err := s.keeper.GetUpgradePlan(newCtx) + require.ErrorIs(t, err, types.ErrNoUpgradePlanFound) } func TestCanClear(t *testing.T) { @@ -529,9 +527,8 @@ func TestDowngradeVerification(t *testing.T) { require.Equal(t, planName, lastAppliedPlan) require.False(t, k.HasHandler(planName)) require.False(t, k.DowngradeVerified()) - _, found, err := k.GetUpgradePlan(ctx) - require.NoError(t, err) - require.False(t, found) + _, err = k.GetUpgradePlan(ctx) + require.ErrorIs(t, err, types.ErrNoUpgradePlanFound) if tc.preRun != nil { tc.preRun(k, ctx, name) diff --git a/x/upgrade/handler.go b/x/upgrade/handler.go index 3c5511c9d809..6e09c39a6bc7 100644 --- a/x/upgrade/handler.go +++ b/x/upgrade/handler.go @@ -38,6 +38,5 @@ func handleSoftwareUpgradeProposal(ctx sdk.Context, k *keeper.Keeper, p *types.S //nolint:staticcheck // we are intentionally using a deprecated proposal here. func handleCancelSoftwareUpgradeProposal(ctx sdk.Context, k *keeper.Keeper, _ *types.CancelSoftwareUpgradeProposal) error { - k.ClearUpgradePlan(ctx) - return nil + return k.ClearUpgradePlan(ctx) } diff --git a/x/upgrade/keeper/grpc_query.go b/x/upgrade/keeper/grpc_query.go index 6b324e586ee6..dacfc2ecae26 100644 --- a/x/upgrade/keeper/grpc_query.go +++ b/x/upgrade/keeper/grpc_query.go @@ -2,12 +2,12 @@ package keeper import ( "context" + "errors" errorsmod "cosmossdk.io/errors" "cosmossdk.io/x/upgrade/types" sdk "github.com/cosmos/cosmos-sdk/types" - "github.com/cosmos/cosmos-sdk/types/errors" ) var _ types.QueryServer = Keeper{} @@ -16,13 +16,13 @@ var _ types.QueryServer = Keeper{} func (k Keeper) CurrentPlan(c context.Context, req *types.QueryCurrentPlanRequest) (*types.QueryCurrentPlanResponse, error) { ctx := sdk.UnwrapSDKContext(c) - plan, found, err := k.GetUpgradePlan(ctx) + plan, err := k.GetUpgradePlan(ctx) if err != nil { - return nil, err - } + if errors.Is(err, types.ErrNoUpgradePlanFound) { + return &types.QueryCurrentPlanResponse{}, nil + } - if !found { - return &types.QueryCurrentPlanResponse{}, nil + return nil, err } return &types.QueryCurrentPlanResponse{Plan: &plan}, nil @@ -41,13 +41,13 @@ func (k Keeper) AppliedPlan(c context.Context, req *types.QueryAppliedPlanReques func (k Keeper) UpgradedConsensusState(c context.Context, req *types.QueryUpgradedConsensusStateRequest) (*types.QueryUpgradedConsensusStateResponse, error) { //nolint:staticcheck // we're using a deprecated call for compatibility ctx := sdk.UnwrapSDKContext(c) - consState, found, err := k.GetUpgradedConsensusState(ctx, req.LastHeight) + consState, err := k.GetUpgradedConsensusState(ctx, req.LastHeight) if err != nil { - return nil, err - } + if errors.Is(err, types.ErrNoUpgradedConsensusStateFound) { + return &types.QueryUpgradedConsensusStateResponse{}, nil + } - if !found { - return &types.QueryUpgradedConsensusStateResponse{}, nil //nolint:staticcheck // we're using a deprecated call for compatibility + return nil, err } return &types.QueryUpgradedConsensusStateResponse{ //nolint:staticcheck // we're using a deprecated call for compatibility @@ -61,18 +61,15 @@ func (k Keeper) ModuleVersions(c context.Context, req *types.QueryModuleVersions // check if a specific module was requested if len(req.ModuleName) > 0 { - version, ok, err := k.getModuleVersion(ctx, req.ModuleName) + version, err := k.getModuleVersion(ctx, req.ModuleName) if err != nil { - return nil, err + // module requested, but not found or error happened + return nil, errorsmod.Wrapf(err, "x/upgrade: QueryModuleVersions module %s not found", req.ModuleName) } - if ok { - // return the requested module - res := []*types.ModuleVersion{{Name: req.ModuleName, Version: version}} - return &types.QueryModuleVersionsResponse{ModuleVersions: res}, nil - } - // module requested, but not found - return nil, errorsmod.Wrapf(errors.ErrNotFound, "x/upgrade: QueryModuleVersions module %s not found", req.ModuleName) + // return the requested module + res := []*types.ModuleVersion{{Name: req.ModuleName, Version: version}} + return &types.QueryModuleVersionsResponse{ModuleVersions: res}, nil } // if no module requested return all module versions from state diff --git a/x/upgrade/keeper/keeper.go b/x/upgrade/keeper/keeper.go index 2dc150b1ef83..6e3d8515fc74 100644 --- a/x/upgrade/keeper/keeper.go +++ b/x/upgrade/keeper/keeper.go @@ -4,6 +4,7 @@ import ( "context" "encoding/binary" "encoding/json" + "errors" "fmt" "os" "path" @@ -201,23 +202,28 @@ func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, } // getModuleVersion gets the version for a given module, and returns true if it exists, false otherwise -func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, bool, error) { +func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, error) { store := k.storeService.OpenKVStore(ctx) prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - return 0, false, err + return 0, err } for ; it.Valid(); it.Next() { moduleName := string(it.Key()[1:]) if moduleName == name { version := binary.BigEndian.Uint64(it.Value()) - return version, true, nil + return version, nil } } - return 0, false, it.Close() + err = it.Close() + if err != nil { + return 0, err + } + + return 0, types.ErrNoModuleVersionFound } // ScheduleUpgrade schedules an upgrade based on the specified plan. @@ -248,12 +254,15 @@ func (k Keeper) ScheduleUpgrade(ctx context.Context, plan types.Plan) error { store := k.storeService.OpenKVStore(ctx) // clear any old IBC state stored by previous plan - oldPlan, found, err := k.GetUpgradePlan(ctx) + oldPlan, err := k.GetUpgradePlan(ctx) + // if there's an error but it's not ErrNoUpgradePlanFound, return error if err != nil { - return err + if !errors.Is(err, types.ErrNoUpgradePlanFound) { + return err + } } - if found { + if err == nil { err = k.ClearIBCState(ctx, oldPlan.Height) if err != nil { return err @@ -282,14 +291,18 @@ func (k Keeper) SetUpgradedClient(ctx context.Context, planHeight int64, bz []by } // GetUpgradedClient gets the expected upgraded client for the next version of this chain -func (k Keeper) GetUpgradedClient(ctx context.Context, height int64) ([]byte, bool, error) { +func (k Keeper) GetUpgradedClient(ctx context.Context, height int64) ([]byte, error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.UpgradedClientKey(height)) - if len(bz) == 0 || err != nil { - return nil, false, err + if err != nil { + return nil, err } - return bz, true, nil + if bz == nil { + return nil, types.ErrNoUpgradedClientFound + } + + return bz, nil } // SetUpgradedConsensusState sets the expected upgraded consensus state for the next version of this chain @@ -299,15 +312,19 @@ func (k Keeper) SetUpgradedConsensusState(ctx context.Context, planHeight int64, return store.Set(types.UpgradedConsStateKey(planHeight), bz) } -// GetUpgradedConsensusState gets the expected upgraded consensus state for the next version of this chain -func (k Keeper) GetUpgradedConsensusState(ctx context.Context, lastHeight int64) ([]byte, bool, error) { +// GetUpgradedConsensusState gets the expected upgraded consensus state for the next version of this chain. +func (k Keeper) GetUpgradedConsensusState(ctx context.Context, lastHeight int64) ([]byte, error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.UpgradedConsStateKey(lastHeight)) - if len(bz) == 0 || err != nil { - return nil, false, err + if err != nil { + return nil, err + } + + if bz == nil { + return nil, types.ErrNoUpgradedConsensusStateFound } - return bz, true, nil + return bz, nil } // GetLastCompletedUpgrade returns the last applied upgrade name and height. @@ -377,17 +394,19 @@ func (k Keeper) ClearIBCState(ctx context.Context, lastHeight int64) error { // ClearUpgradePlan clears any schedule upgrade and associated IBC states. func (k Keeper) ClearUpgradePlan(ctx context.Context) error { - // clear IBC states everytime upgrade plan is removed - oldPlan, found, err := k.GetUpgradePlan(ctx) + // clear IBC states every time upgrade plan is removed + oldPlan, err := k.GetUpgradePlan(ctx) if err != nil { + // if there's no upgrade plan, return nil to match previous behavior + if errors.Is(err, types.ErrNoUpgradePlanFound) { + return nil + } return err } - if found { - err := k.ClearIBCState(ctx, oldPlan.Height) - if err != nil { - return err - } + err = k.ClearIBCState(ctx, oldPlan.Height) + if err != nil { + return err } store := k.storeService.OpenKVStore(ctx) @@ -400,21 +419,25 @@ func (k Keeper) Logger(ctx context.Context) log.Logger { return sdkCtx.Logger().With("module", "x/"+types.ModuleName) } -// GetUpgradePlan returns the currently scheduled Plan if any, setting havePlan to true if there is a scheduled -// upgrade or false if there is none -func (k Keeper) GetUpgradePlan(ctx context.Context) (plan types.Plan, havePlan bool, err error) { +// GetUpgradePlan returns the currently scheduled Plan if any, setting err to nil if there is a scheduled +// upgrade or to ErrNoUpgradePlanFound if there is none. If err is any other than ErrNoUpgradePlanFound, an error occurred. +func (k Keeper) GetUpgradePlan(ctx context.Context) (plan types.Plan, err error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.PlanKey()) - if bz == nil || err != nil { - return plan, false, err + if err != nil { + return plan, err + } + + if bz == nil { + return plan, types.ErrNoUpgradePlanFound } err = k.cdc.Unmarshal(bz, &plan) if err != nil { - return plan, false, err + return plan, err } - return plan, true, err + return plan, err } // setDone marks this upgrade name as being done so the name can't be reused accidentally diff --git a/x/upgrade/keeper/keeper_test.go b/x/upgrade/keeper/keeper_test.go index 2f8f0ecb3639..802685b5b3a5 100644 --- a/x/upgrade/keeper/keeper_test.go +++ b/x/upgrade/keeper/keeper_test.go @@ -190,13 +190,13 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { name string height int64 setup func() - exists bool + err bool }{ { name: "no upgraded client exists", height: 10, setup: func() {}, - exists: false, + err: true, }, { name: "success", @@ -204,7 +204,7 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { setup: func() { s.upgradeKeeper.SetUpgradedClient(s.ctx, 10, cs) }, - exists: true, + err: false, }, } @@ -215,15 +215,14 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { // setup test case tc.setup() - gotCs, exists, err := s.upgradeKeeper.GetUpgradedClient(s.ctx, tc.height) - s.Require().NoError(err) + gotCs, err := s.upgradeKeeper.GetUpgradedClient(s.ctx, tc.height) - if tc.exists { - s.Require().Equal(cs, gotCs, "valid case: %s did not retrieve correct client state", tc.name) - s.Require().True(exists, "valid case: %s did not retrieve client state", tc.name) - } else { + if tc.err { s.Require().Nil(gotCs, "invalid case: %s retrieved valid client state", tc.name) - s.Require().False(exists, "invalid case: %s retrieved valid client state", tc.name) + s.Require().Error(err, "invalid case: %s retrieved valid client state", tc.name) + } else { + s.Require().Equal(cs, gotCs, "valid case: %s did not retrieve correct client state", tc.name) + s.Require().NoError(err, "valid case: %s did not retrieve client state", tc.name) } } } @@ -243,8 +242,7 @@ func (s *KeeperTestSuite) TestIsSkipHeight() { func (s *KeeperTestSuite) TestUpgradedConsensusState() { cs := []byte("IBC consensus state") s.Require().NoError(s.upgradeKeeper.SetUpgradedConsensusState(s.ctx, 10, cs)) - bz, ok, err := s.upgradeKeeper.GetUpgradedConsensusState(s.ctx, 10) - s.Require().True(ok) + bz, err := s.upgradeKeeper.GetUpgradedConsensusState(s.ctx, 10) s.Require().Equal(cs, bz) s.Require().NoError(err) } diff --git a/x/upgrade/keeper/msg_server.go b/x/upgrade/keeper/msg_server.go index dcf5900b6e24..7c8cc8a48c35 100644 --- a/x/upgrade/keeper/msg_server.go +++ b/x/upgrade/keeper/msg_server.go @@ -50,7 +50,10 @@ func (k msgServer) CancelUpgrade(ctx context.Context, msg *types.MsgCancelUpgrad } sdkCtx := sdk.UnwrapSDKContext(ctx) - k.ClearUpgradePlan(sdkCtx) + err := k.ClearUpgradePlan(sdkCtx) + if err != nil { + return nil, err + } return &types.MsgCancelUpgradeResponse{}, nil } diff --git a/x/upgrade/keeper/msg_server_test.go b/x/upgrade/keeper/msg_server_test.go index 5647b81db873..8c04b328fa33 100644 --- a/x/upgrade/keeper/msg_server_test.go +++ b/x/upgrade/keeper/msg_server_test.go @@ -74,11 +74,9 @@ func (s *KeeperTestSuite) TestSoftwareUpgrade() { s.Require().Contains(err.Error(), tc.errMsg) } else { s.Require().NoError(err) - plan, found, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) + plan, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) s.Require().NoError(err) - s.Require().Equal(true, found) s.Require().Equal(tc.req.Plan, plan) - } }) } @@ -133,9 +131,8 @@ func (s *KeeperTestSuite) TestCancelUpgrade() { s.Require().Contains(err.Error(), tc.errMsg) } else { s.Require().NoError(err) - _, found, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) - s.Require().NoError(err) - s.Require().Equal(false, found) + _, err := s.upgradeKeeper.GetUpgradePlan(s.ctx) + s.Require().ErrorIs(err, types.ErrNoUpgradePlanFound) } }) } From 7507d422ab4ca173b3192e48685e1f47e539b4df Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 10:30:08 +0200 Subject: [PATCH 17/24] handle not founds with errors --- x/upgrade/types/errors.go | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) create mode 100644 x/upgrade/types/errors.go diff --git a/x/upgrade/types/errors.go b/x/upgrade/types/errors.go new file mode 100644 index 000000000000..e1877a1acf20 --- /dev/null +++ b/x/upgrade/types/errors.go @@ -0,0 +1,17 @@ +package types + +import ( + "cosmossdk.io/errors" +) + +// x/authz module sentinel errors +var ( + // ErrNoModuleVersionFound error if there is no version found in the module-version map + ErrNoModuleVersionFound = errors.Register(ModuleName, 2, "module version not found") + // ErrNoUpgradePlanFound error if there is no scheduled upgrade plan found + ErrNoUpgradePlanFound = errors.Register(ModuleName, 3, "upgrade plan not found") + // ErrNoUpgradedClientFound error if there is no upgraded client for the next version + ErrNoUpgradedClientFound = errors.Register(ModuleName, 4, "upgraded client not found") + // ErrNoUpgradedConsensusStateFound error if there is no upgraded consensus state for the next version + ErrNoUpgradedConsensusStateFound = errors.Register(ModuleName, 5, "upgraded consensus state not found") +) From 21fdc52098745d6dcfb2ae5cc904ff185a16c925 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 10:48:47 +0200 Subject: [PATCH 18/24] fix comments --- x/upgrade/keeper/keeper.go | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/x/upgrade/keeper/keeper.go b/x/upgrade/keeper/keeper.go index 6e3d8515fc74..cb169c683d3b 100644 --- a/x/upgrade/keeper/keeper.go +++ b/x/upgrade/keeper/keeper.go @@ -201,7 +201,8 @@ func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, return mv, it.Close() } -// getModuleVersion gets the version for a given module, and returns true if it exists, false otherwise +// getModuleVersion gets the version for a given module. If it doesn't exist it returns ErrNoModuleVersionFound, other +// errors may be returned if there is an error reading from the store. func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, error) { store := k.storeService.OpenKVStore(ctx) prefix := []byte{types.VersionMapByte} @@ -256,10 +257,8 @@ func (k Keeper) ScheduleUpgrade(ctx context.Context, plan types.Plan) error { // clear any old IBC state stored by previous plan oldPlan, err := k.GetUpgradePlan(ctx) // if there's an error but it's not ErrNoUpgradePlanFound, return error - if err != nil { - if !errors.Is(err, types.ErrNoUpgradePlanFound) { - return err - } + if err != nil && !errors.Is(err, types.ErrNoUpgradePlanFound) { + return err } if err == nil { @@ -290,7 +289,8 @@ func (k Keeper) SetUpgradedClient(ctx context.Context, planHeight int64, bz []by return store.Set(types.UpgradedClientKey(planHeight), bz) } -// GetUpgradedClient gets the expected upgraded client for the next version of this chain +// GetUpgradedClient gets the expected upgraded client for the next version of this chain. If not found it returns +// ErrNoUpgradedClientFound, but other errors may be returned if there is an error reading from the store. func (k Keeper) GetUpgradedClient(ctx context.Context, height int64) ([]byte, error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.UpgradedClientKey(height)) @@ -312,7 +312,8 @@ func (k Keeper) SetUpgradedConsensusState(ctx context.Context, planHeight int64, return store.Set(types.UpgradedConsStateKey(planHeight), bz) } -// GetUpgradedConsensusState gets the expected upgraded consensus state for the next version of this chain. +// GetUpgradedConsensusState gets the expected upgraded consensus state for the next version of this chain. If not found +// it returns ErrNoUpgradedConsensusStateFound, but other errors may be returned if there is an error reading from the store. func (k Keeper) GetUpgradedConsensusState(ctx context.Context, lastHeight int64) ([]byte, error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.UpgradedConsStateKey(lastHeight)) @@ -419,8 +420,8 @@ func (k Keeper) Logger(ctx context.Context) log.Logger { return sdkCtx.Logger().With("module", "x/"+types.ModuleName) } -// GetUpgradePlan returns the currently scheduled Plan if any, setting err to nil if there is a scheduled -// upgrade or to ErrNoUpgradePlanFound if there is none. If err is any other than ErrNoUpgradePlanFound, an error occurred. +// GetUpgradePlan returns the currently scheduled Plan if any. If not found it returns +// ErrNoUpgradePlanFound, but other errors may be returned if there is an error reading from the store. func (k Keeper) GetUpgradePlan(ctx context.Context) (plan types.Plan, err error) { store := k.storeService.OpenKVStore(ctx) bz, err := store.Get(types.PlanKey()) From 559568e576b8ad0305272120e0b61ca0f455d0a1 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 10:49:26 +0200 Subject: [PATCH 19/24] lint --- x/upgrade/keeper/grpc_query.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x/upgrade/keeper/grpc_query.go b/x/upgrade/keeper/grpc_query.go index dacfc2ecae26..d3502372e731 100644 --- a/x/upgrade/keeper/grpc_query.go +++ b/x/upgrade/keeper/grpc_query.go @@ -44,7 +44,7 @@ func (k Keeper) UpgradedConsensusState(c context.Context, req *types.QueryUpgrad consState, err := k.GetUpgradedConsensusState(ctx, req.LastHeight) if err != nil { if errors.Is(err, types.ErrNoUpgradedConsensusStateFound) { - return &types.QueryUpgradedConsensusStateResponse{}, nil + return &types.QueryUpgradedConsensusStateResponse{}, nil //nolint:staticcheck // we're using a deprecated call for compatibility } return nil, err From 8d769ce7f88b7272035088f4e33d8cbb76bfc216 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 10:55:12 +0200 Subject: [PATCH 20/24] update changelog --- x/upgrade/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x/upgrade/CHANGELOG.md b/x/upgrade/CHANGELOG.md index 17fe463ed82c..78c064a6f508 100644 --- a/x/upgrade/CHANGELOG.md +++ b/x/upgrade/CHANGELOG.md @@ -32,4 +32,4 @@ Ref: https://keepachangelog.com/en/1.0.0/ ### API Breaking -* (x/upgrade) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error`. Also `UpgradeHandler` now receives a `context.Context`. +* (x/upgrade) [#16227](https://github.com/cosmos/cosmos-sdk/issues/16227) `NewKeeper` now takes a `KVStoreService` instead of a `StoreKey`, methods in the `Keeper` now take a `context.Context` instead of a `sdk.Context` and return an `error`. `UpgradeHandler` now receives a `context.Context`. `GetUpgradedClient`, `GetUpgradedConsensusState`, `GetUpgradePlan` now return a specific error for "not found". From d310db3b9add60ffd9c3ede5c0259f3b6842ed53 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 11:13:36 +0200 Subject: [PATCH 21/24] close all iterators --- x/upgrade/keeper/keeper.go | 8 +++++++- x/upgrade/keeper/migrations.go | 3 +-- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/x/upgrade/keeper/keeper.go b/x/upgrade/keeper/keeper.go index cb169c683d3b..0724e31623ef 100644 --- a/x/upgrade/keeper/keeper.go +++ b/x/upgrade/keeper/keeper.go @@ -163,6 +163,7 @@ func (k Keeper) GetModuleVersionMap(ctx context.Context) (module.VersionMap, err prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { + _ = it.Close() return nil, err } @@ -184,6 +185,7 @@ func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { + _ = it.Close() return nil, err } @@ -208,6 +210,7 @@ func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, erro prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { + _ = it.Close() return 0, err } @@ -215,7 +218,8 @@ func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, erro moduleName := string(it.Key()[1:]) if moduleName == name { version := binary.BigEndian.Uint64(it.Value()) - return version, nil + err = it.Close() + return version, err } } @@ -334,6 +338,7 @@ func (k Keeper) GetLastCompletedUpgrade(ctx context.Context) (string, int64, err prefix := []byte{types.DoneByte} it, err := store.ReverseIterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { + _ = it.Close() return "", 0, err } @@ -368,6 +373,7 @@ func (k Keeper) GetDoneHeight(ctx context.Context, name string) (int64, error) { prefix := []byte{types.DoneByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { + _ = it.Close() return 0, err } diff --git a/x/upgrade/keeper/migrations.go b/x/upgrade/keeper/migrations.go index 3f210f57673a..53ef4933e879 100644 --- a/x/upgrade/keeper/migrations.go +++ b/x/upgrade/keeper/migrations.go @@ -3,9 +3,8 @@ package keeper import ( "encoding/binary" - "cosmossdk.io/store/prefix" - // storetypes "cosmossdk.io/store/types" storetypes "cosmossdk.io/core/store" + "cosmossdk.io/store/prefix" "cosmossdk.io/x/upgrade/types" "github.com/cosmos/cosmos-sdk/runtime" From 53a50c6d7c473bdb25c253c675dceb43a74468aa Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 11:20:00 +0200 Subject: [PATCH 22/24] defer all iterators --- x/upgrade/keeper/keeper.go | 30 ++++++++++++------------------ 1 file changed, 12 insertions(+), 18 deletions(-) diff --git a/x/upgrade/keeper/keeper.go b/x/upgrade/keeper/keeper.go index 0724e31623ef..eae4cbdef44b 100644 --- a/x/upgrade/keeper/keeper.go +++ b/x/upgrade/keeper/keeper.go @@ -163,9 +163,9 @@ func (k Keeper) GetModuleVersionMap(ctx context.Context) (module.VersionMap, err prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - _ = it.Close() return nil, err } + defer it.Close() vm := make(module.VersionMap) for ; it.Valid(); it.Next() { @@ -176,7 +176,7 @@ func (k Keeper) GetModuleVersionMap(ctx context.Context) (module.VersionMap, err vm[name] = moduleVersion } - return vm, it.Close() + return vm, nil } // GetModuleVersions gets a slice of module consensus versions @@ -185,9 +185,9 @@ func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - _ = it.Close() return nil, err } + defer it.Close() mv := make([]*types.ModuleVersion, 0) for ; it.Valid(); it.Next() { @@ -200,7 +200,7 @@ func (k Keeper) GetModuleVersions(ctx context.Context) ([]*types.ModuleVersion, }) } - return mv, it.Close() + return mv, nil } // getModuleVersion gets the version for a given module. If it doesn't exist it returns ErrNoModuleVersionFound, other @@ -210,24 +210,18 @@ func (k Keeper) getModuleVersion(ctx context.Context, name string) (uint64, erro prefix := []byte{types.VersionMapByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - _ = it.Close() return 0, err } + defer it.Close() for ; it.Valid(); it.Next() { moduleName := string(it.Key()[1:]) if moduleName == name { version := binary.BigEndian.Uint64(it.Value()) - err = it.Close() - return version, err + return version, nil } } - err = it.Close() - if err != nil { - return 0, err - } - return 0, types.ErrNoModuleVersionFound } @@ -338,16 +332,16 @@ func (k Keeper) GetLastCompletedUpgrade(ctx context.Context) (string, int64, err prefix := []byte{types.DoneByte} it, err := store.ReverseIterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - _ = it.Close() return "", 0, err } + defer it.Close() if it.Valid() { name, height := parseDoneKey(it.Key()) - return name, height, it.Close() + return name, height, nil } - return "", 0, it.Close() + return "", 0, nil } // parseDoneKey - split upgrade name and height from the done key @@ -373,18 +367,18 @@ func (k Keeper) GetDoneHeight(ctx context.Context, name string) (int64, error) { prefix := []byte{types.DoneByte} it, err := store.Iterator(prefix, storetypes.PrefixEndBytes(prefix)) if err != nil { - _ = it.Close() return 0, err } + defer it.Close() for ; it.Valid(); it.Next() { upgradeName, height := parseDoneKey(it.Key()) if upgradeName == name { - return height, it.Close() + return height, nil } } - return 0, it.Close() + return 0, nil } // ClearIBCState clears any planned IBC state From f7a9bf6d9992d50a8a097cf4464a388270a832c4 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 22 May 2023 11:52:24 +0200 Subject: [PATCH 23/24] fix juliens suggestion --- x/upgrade/keeper/keeper_test.go | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/x/upgrade/keeper/keeper_test.go b/x/upgrade/keeper/keeper_test.go index 802685b5b3a5..ab150e8ff838 100644 --- a/x/upgrade/keeper/keeper_test.go +++ b/x/upgrade/keeper/keeper_test.go @@ -190,13 +190,13 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { name string height int64 setup func() - err bool + exists bool }{ { name: "no upgraded client exists", height: 10, setup: func() {}, - err: true, + exists: false, }, { name: "success", @@ -204,7 +204,7 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { setup: func() { s.upgradeKeeper.SetUpgradedClient(s.ctx, 10, cs) }, - err: false, + exists: true, }, } @@ -217,12 +217,12 @@ func (s *KeeperTestSuite) TestSetUpgradedClient() { gotCs, err := s.upgradeKeeper.GetUpgradedClient(s.ctx, tc.height) - if tc.err { - s.Require().Nil(gotCs, "invalid case: %s retrieved valid client state", tc.name) - s.Require().Error(err, "invalid case: %s retrieved valid client state", tc.name) - } else { + if tc.exists { s.Require().Equal(cs, gotCs, "valid case: %s did not retrieve correct client state", tc.name) s.Require().NoError(err, "valid case: %s did not retrieve client state", tc.name) + } else { + s.Require().Nil(gotCs, "invalid case: %s retrieved valid client state", tc.name) + s.Require().Error(err, "invalid case: %s retrieved valid client state", tc.name) } } } From f397d9ca36a913d0dd04d974e574804edf99c195 Mon Sep 17 00:00:00 2001 From: Facundo Medica Date: Mon, 5 Jun 2023 15:22:29 +0200 Subject: [PATCH 24/24] fix test --- x/upgrade/abci_test.go | 3 +-- x/upgrade/types/plan_test.go | 2 -- 2 files changed, 1 insertion(+), 4 deletions(-) diff --git a/x/upgrade/abci_test.go b/x/upgrade/abci_test.go index 8b175d7c3305..0a9f31761705 100644 --- a/x/upgrade/abci_test.go +++ b/x/upgrade/abci_test.go @@ -11,7 +11,6 @@ import ( "cosmossdk.io/core/header" "cosmossdk.io/log" storetypes "cosmossdk.io/store/types" - cmtproto "github.com/cometbft/cometbft/proto/tendermint/types" "github.com/stretchr/testify/require" "github.com/stretchr/testify/suite" @@ -154,7 +153,7 @@ func TestHaltIfTooNew(t *testing.T) { require.Equal(t, 0, called) t.Log("Verify we error if we have a registered handler ahead of time") - err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop", Plan: types.Plan{Name: "future", Height: s.ctx.BlockHeight() + 3}}) //nolint:staticcheck // we're testing deprecated code + err = s.handler(s.ctx, &types.SoftwareUpgradeProposal{Title: "prop", Plan: types.Plan{Name: "future", Height: s.ctx.HeaderInfo().Height + 3}}) //nolint:staticcheck // we're testing deprecated code require.NoError(t, err) err = s.module.BeginBlock(newCtx) diff --git a/x/upgrade/types/plan_test.go b/x/upgrade/types/plan_test.go index 9157d2d4d3b4..423f12b1d47f 100644 --- a/x/upgrade/types/plan_test.go +++ b/x/upgrade/types/plan_test.go @@ -7,8 +7,6 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" - "cosmossdk.io/core/header" - "cosmossdk.io/log" "cosmossdk.io/x/upgrade/types" codectypes "github.com/cosmos/cosmos-sdk/codec/types"