From 34abe313ad476393b51457a4c7b2091f45ca689d Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Fri, 22 Nov 2024 14:36:41 +0100 Subject: [PATCH 1/6] feat(photon): store upgrade --- app/app.go | 3 +- app/upgrades/v2/constants.go | 23 +++++++++++++ app/upgrades/v2/upgrades.go | 66 ++++++++++++++++++++++++++++++++++++ 3 files changed, 91 insertions(+), 1 deletion(-) create mode 100644 app/upgrades/v2/constants.go create mode 100644 app/upgrades/v2/upgrades.go diff --git a/app/app.go b/app/app.go index 9a8261a8..33439649 100644 --- a/app/app.go +++ b/app/app.go @@ -51,6 +51,7 @@ import ( "github.com/atomone-hub/atomone/app/keepers" "github.com/atomone-hub/atomone/app/params" "github.com/atomone-hub/atomone/app/upgrades" + v2 "github.com/atomone-hub/atomone/app/upgrades/v2" govtypes "github.com/atomone-hub/atomone/x/gov/types" ) @@ -58,7 +59,7 @@ var ( // DefaultNodeHome default home directories for the application daemon DefaultNodeHome string - Upgrades = []upgrades.Upgrade{} + Upgrades = []upgrades.Upgrade{v2.Upgrade} ) var ( diff --git a/app/upgrades/v2/constants.go b/app/upgrades/v2/constants.go new file mode 100644 index 00000000..2b184a3d --- /dev/null +++ b/app/upgrades/v2/constants.go @@ -0,0 +1,23 @@ +package v2 + +import ( + store "github.com/cosmos/cosmos-sdk/store/types" + + "github.com/atomone-hub/atomone/app/upgrades" + photontypes "github.com/atomone-hub/atomone/x/photon/types" +) + +const ( + UpgradeName = "v2" +) + +var Upgrade = upgrades.Upgrade{ + UpgradeName: UpgradeName, + CreateUpgradeHandler: CreateUpgradeHandler, + StoreUpgrades: store.StoreUpgrades{ + Added: []string{ + // new module added in v2 + photontypes.ModuleName, + }, + }, +} diff --git a/app/upgrades/v2/upgrades.go b/app/upgrades/v2/upgrades.go new file mode 100644 index 00000000..23872895 --- /dev/null +++ b/app/upgrades/v2/upgrades.go @@ -0,0 +1,66 @@ +package v2 + +import ( + sdk "github.com/cosmos/cosmos-sdk/types" + "github.com/cosmos/cosmos-sdk/types/module" + bankkeeper "github.com/cosmos/cosmos-sdk/x/bank/keeper" + banktypes "github.com/cosmos/cosmos-sdk/x/bank/types" + upgradetypes "github.com/cosmos/cosmos-sdk/x/upgrade/types" + + "github.com/atomone-hub/atomone/app/keepers" +) + +// CreateUpgradeHandler returns a upgrade handler for AtomOne v2 +// which executes the following migrations: +// - add new denom metadata for photon in the bank module store. +func CreateUpgradeHandler( + mm *module.Manager, + configurator module.Configurator, + keepers *keepers.AppKeepers, +) upgradetypes.UpgradeHandler { + return func(ctx sdk.Context, plan upgradetypes.Plan, vm module.VersionMap) (module.VersionMap, error) { + ctx.Logger().Info("Starting module migrations...") + // Admitedly there's no module migration because v2 doesn't upgrade the + // SDK, but still running it for demo purpose. + vm, err := mm.RunMigrations(ctx, configurator, vm) + if err != nil { + return vm, err + } + // Add the photon denom metadata to the bank module store + setPhotonDenomMetadata(ctx, keepers.BankKeeper) + ctx.Logger().Info("Upgrade complete") + return vm, nil + } +} + +func setPhotonDenomMetadata(ctx sdk.Context, bk bankkeeper.Keeper) { + ctx.Logger().Info("Adding photon denom metadata...") + bk.SetDenomMetaData(ctx, banktypes.Metadata{ + Base: "uphoton", + Display: "photon", + Name: "AtomOne Photon", + Symbol: "PHOTON", + Description: "The fee token of AtomOne Hub", + DenomUnits: []*banktypes.DenomUnit{ + { + Denom: "uphoton", + Exponent: 0, + Aliases: []string{ + "microphoton", + }, + }, + { + Denom: "mphoton", + Exponent: 3, + Aliases: []string{ + "milliphoton", + }, + }, + { + Denom: "photon", + Exponent: 6, + }, + }, + }) + ctx.Logger().Info("Photon denom metadata added") +} From 8053040bc2c2501d2a8173acb4a0f5299f65e6b0 Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Mon, 25 Nov 2024 10:57:22 +0100 Subject: [PATCH 2/6] fix lint --- app/upgrades/v2/upgrades.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/app/upgrades/v2/upgrades.go b/app/upgrades/v2/upgrades.go index 23872895..83c61f4f 100644 --- a/app/upgrades/v2/upgrades.go +++ b/app/upgrades/v2/upgrades.go @@ -20,8 +20,9 @@ func CreateUpgradeHandler( ) upgradetypes.UpgradeHandler { return func(ctx sdk.Context, plan upgradetypes.Plan, vm module.VersionMap) (module.VersionMap, error) { ctx.Logger().Info("Starting module migrations...") - // Admitedly there's no module migration because v2 doesn't upgrade the - // SDK, but still running it for demo purpose. + // Admittedly there's no module migration because v2 doesn't upgrade the + // SDK nor the internal gov module, but we still running it for demo + // purpose. vm, err := mm.RunMigrations(ctx, configurator, vm) if err != nil { return vm, err From 8ad12a324eea6278c457b82a57d2d516c1cf80f5 Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Mon, 25 Nov 2024 11:15:41 +0100 Subject: [PATCH 3/6] correct comment --- app/upgrades/v2/upgrades.go | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/app/upgrades/v2/upgrades.go b/app/upgrades/v2/upgrades.go index 83c61f4f..95930246 100644 --- a/app/upgrades/v2/upgrades.go +++ b/app/upgrades/v2/upgrades.go @@ -20,9 +20,8 @@ func CreateUpgradeHandler( ) upgradetypes.UpgradeHandler { return func(ctx sdk.Context, plan upgradetypes.Plan, vm module.VersionMap) (module.VersionMap, error) { ctx.Logger().Info("Starting module migrations...") - // Admittedly there's no module migration because v2 doesn't upgrade the - // SDK nor the internal gov module, but we still running it for demo - // purpose. + // RunMigrations will detect the add of the photon module, will initiate + // its genesis and will fill the versionMap with its consensus version. vm, err := mm.RunMigrations(ctx, configurator, vm) if err != nil { return vm, err From c3ce7d8f17dc8a12440ca29d9a696205767ac659 Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Mon, 25 Nov 2024 12:14:11 +0100 Subject: [PATCH 4/6] add FIXME --- x/photon/keeper/msg_server.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x/photon/keeper/msg_server.go b/x/photon/keeper/msg_server.go index 7b90a0dd..2483c07e 100644 --- a/x/photon/keeper/msg_server.go +++ b/x/photon/keeper/msg_server.go @@ -45,7 +45,7 @@ func (k msgServer) MintPhoton(goCtx context.Context, msg *types.MsgMintPhoton) ( uphotonToMint = bondDenomToBurn.Amount.ToLegacyDec().Mul(conversionRate) ) // If no photon to mint, do not burn bondDenomToBurn, returns an error - if uphotonToMint.IsZero() { + if uphotonToMint.IsZero() { // FIXME should be checked against uphotonToMint.RoundInt() bc else coinsToMint can be empty return nil, types.ErrNoMintablePhotons } // If photonToMint + photonSupply > photonMaxSupply, returns an error From a46512587bc00c40779db3023d0b3b6d7030b2e8 Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Mon, 25 Nov 2024 14:31:14 +0100 Subject: [PATCH 5/6] FIXME handled at 8e1732 --- x/photon/keeper/msg_server.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x/photon/keeper/msg_server.go b/x/photon/keeper/msg_server.go index 2483c07e..7b90a0dd 100644 --- a/x/photon/keeper/msg_server.go +++ b/x/photon/keeper/msg_server.go @@ -45,7 +45,7 @@ func (k msgServer) MintPhoton(goCtx context.Context, msg *types.MsgMintPhoton) ( uphotonToMint = bondDenomToBurn.Amount.ToLegacyDec().Mul(conversionRate) ) // If no photon to mint, do not burn bondDenomToBurn, returns an error - if uphotonToMint.IsZero() { // FIXME should be checked against uphotonToMint.RoundInt() bc else coinsToMint can be empty + if uphotonToMint.IsZero() { return nil, types.ErrNoMintablePhotons } // If photonToMint + photonSupply > photonMaxSupply, returns an error From eab976cbfb11068a699f1dd17f8670d7286b14ea Mon Sep 17 00:00:00 2001 From: Thomas Bruyelle Date: Mon, 25 Nov 2024 14:29:03 +0100 Subject: [PATCH 6/6] fix(x/photon): handle when minted uphoton=0 --- x/photon/keeper/msg_server.go | 8 ++++---- x/photon/keeper/msg_server_test.go | 18 +++++++++++++++++- x/photon/types/errors.go | 13 ++++++------- 3 files changed, 27 insertions(+), 12 deletions(-) diff --git a/x/photon/keeper/msg_server.go b/x/photon/keeper/msg_server.go index 7b90a0dd..40a32cd2 100644 --- a/x/photon/keeper/msg_server.go +++ b/x/photon/keeper/msg_server.go @@ -42,14 +42,14 @@ func (k msgServer) MintPhoton(goCtx context.Context, msg *types.MsgMintPhoton) ( uphotonSupply = k.bankKeeper.GetSupply(ctx, types.Denom).Amount.ToLegacyDec() conversionRate = k.conversionRate(ctx, bondDenomSupply, uphotonSupply) bondDenomToBurn = msg.Amount - uphotonToMint = bondDenomToBurn.Amount.ToLegacyDec().Mul(conversionRate) + uphotonToMint = bondDenomToBurn.Amount.ToLegacyDec().Mul(conversionRate).RoundInt() ) // If no photon to mint, do not burn bondDenomToBurn, returns an error if uphotonToMint.IsZero() { - return nil, types.ErrNoMintablePhotons + return nil, types.ErrZeroMintPhotons } // If photonToMint + photonSupply > photonMaxSupply, returns an error - if uphotonSupply.Add(uphotonToMint).GT(sdk.NewDec(types.MaxSupply)) { + if uphotonSupply.Add(uphotonToMint.ToLegacyDec()).GT(sdk.NewDec(types.MaxSupply)) { return nil, types.ErrNotEnoughPhotons } @@ -60,7 +60,7 @@ func (k msgServer) MintPhoton(goCtx context.Context, msg *types.MsgMintPhoton) ( // 4) move PHOTONs from this module address to msg signer address var ( coinsToBurn = sdk.NewCoins(bondDenomToBurn) - coinsToMint = sdk.NewCoins(sdk.NewCoin(types.Denom, uphotonToMint.RoundInt())) + coinsToMint = sdk.NewCoins(sdk.NewCoin(types.Denom, uphotonToMint)) ) // 1) Send atone to photon module for burn to, err := sdk.AccAddressFromBech32(msg.ToAddress) diff --git a/x/photon/keeper/msg_server_test.go b/x/photon/keeper/msg_server_test.go index 4db659fb..86415910 100644 --- a/x/photon/keeper/msg_server_test.go +++ b/x/photon/keeper/msg_server_test.go @@ -1,6 +1,7 @@ package keeper_test import ( + "math" "testing" "github.com/stretchr/testify/require" @@ -64,7 +65,7 @@ func TestMsgServerMintPhoton(t *testing.T) { Return(sdk.NewInt64Coin(appparams.BondDenom, atoneSupply)) m.BankKeeper.EXPECT().GetSupply(ctx, types.Denom).Return(sdk.NewInt64Coin(types.Denom, types.MaxSupply)) }, - expectedErr: "no more photon can be minted", + expectedErr: "no mintable photon after rounding, try higher burn", }, { name: "fail: photon_supply+minted>max", @@ -81,6 +82,21 @@ func TestMsgServerMintPhoton(t *testing.T) { }, expectedErr: "not enough photon can be minted", }, + { + name: "fail: atone_supply >> photon_supply", + params: types.Params{MintDisabled: false}, + msg: &types.MsgMintPhoton{ + ToAddress: toAddress.String(), + Amount: sdk.NewInt64Coin(appparams.BondDenom, 1), + }, + setup: func(ctx sdk.Context, m testutil.Mocks) { + m.StakingKeeper.EXPECT().BondDenom(ctx).Return(appparams.BondDenom) + m.BankKeeper.EXPECT().GetSupply(ctx, appparams.BondDenom). + Return(sdk.NewInt64Coin(appparams.BondDenom, math.MaxInt)) + m.BankKeeper.EXPECT().GetSupply(ctx, types.Denom).Return(sdk.NewInt64Coin(types.Denom, 0)) + }, + expectedErr: "no mintable photon after rounding, try higher burn", + }, { name: "ok: photon_supply=0", params: types.Params{MintDisabled: false}, diff --git a/x/photon/types/errors.go b/x/photon/types/errors.go index 2cfd7386..031b7eb1 100644 --- a/x/photon/types/errors.go +++ b/x/photon/types/errors.go @@ -6,11 +6,10 @@ import ( // x/photon module sentinel errors var ( - ErrMintDisabled = sdkerrors.Register(ModuleName, 1, "photon mint disabled") //nolint:staticcheck - ErrBurnInvalidDenom = sdkerrors.Register(ModuleName, 2, "invalid burned amount denom: expected bond denom") //nolint:staticcheck - ErrNoMintablePhotons = sdkerrors.Register(ModuleName, 3, "no more photon can be minted") //nolint:staticcheck - ErrNotEnoughPhotons = sdkerrors.Register(ModuleName, 4, "not enough photon can be minted") //nolint:staticcheck - ErrTooManyFeeCoins = sdkerrors.Register(ModuleName, 5, "too many fee coins, only accepts fees in one denom") //nolint:staticcheck - ErrInvalidFeeToken = sdkerrors.Register(ModuleName, 6, "invalid fee token") //nolint:staticcheck - + ErrMintDisabled = sdkerrors.Register(ModuleName, 1, "photon mint disabled") //nolint:staticcheck + ErrBurnInvalidDenom = sdkerrors.Register(ModuleName, 2, "invalid burned amount denom: expected bond denom") //nolint:staticcheck + ErrZeroMintPhotons = sdkerrors.Register(ModuleName, 3, "no mintable photon after rounding, try higher burn") //nolint:staticcheck + ErrNotEnoughPhotons = sdkerrors.Register(ModuleName, 4, "not enough photon can be minted") //nolint:staticcheck + ErrTooManyFeeCoins = sdkerrors.Register(ModuleName, 5, "too many fee coins, only accepts fees in one denom") //nolint:staticcheck + ErrInvalidFeeToken = sdkerrors.Register(ModuleName, 6, "invalid fee token") //nolint:staticcheck )