Skip to content

Commit

Permalink
reward collector - ICA message unit tests (#1005)
Browse files Browse the repository at this point in the history
  • Loading branch information
sampocs authored Dec 4, 2023
1 parent 4fac08f commit fe21ad3
Show file tree
Hide file tree
Showing 4 changed files with 436 additions and 48 deletions.
7 changes: 7 additions & 0 deletions app/apptesting/test_helpers.go
Original file line number Diff line number Diff line change
Expand Up @@ -409,6 +409,13 @@ func (s *AppTestHelper) GetIBCDenomTrace(denom string) transfertypes.DenomTrace
return transfertypes.ParseDenomTrace(prefixedDenom)
}

// Helper function to get the next sequence number for testing when an ICA was submitted
func (s *AppTestHelper) MustGetNextSequenceNumber(portId, channelId string) uint64 {
sequence, found := s.App.StakeibcKeeper.IBCKeeper.ChannelKeeper.GetNextSequenceSend(s.Ctx, portId, channelId)
s.Require().True(found, "sequence number for port %s and channel %s was not found", portId, channelId)
return sequence
}

// Creates and stores an IBC denom from a base denom on transfer channel-0
// This is only required for tests that use the transfer keeper and require that the IBC
// denom is present in the store
Expand Down
19 changes: 12 additions & 7 deletions x/stakeibc/keeper/keeper_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,13 @@ package keeper_test

import (
"testing"
"time"

sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/stretchr/testify/suite"

"github.com/Stride-Labs/stride/v16/app/apptesting"
epochtypes "github.com/Stride-Labs/stride/v16/x/epochs/types"
"github.com/Stride-Labs/stride/v16/x/stakeibc/keeper"
"github.com/Stride-Labs/stride/v16/x/stakeibc/types"
)
Expand All @@ -25,16 +27,9 @@ const (
OsmoPrefix = "osmo"
OsmoChainId = "OSMO"

HostDenom = "udenom"
RewardDenom = "ureward"

ValAddress = "cosmosvaloper1uk4ze0x4nvh4fk0xm4jdud58eqn4yxhrdt795p"
HostICAAddress = "cosmos1gcx4yeplccq9nk6awzmm0gq8jf7yet80qj70tkwy0mz7pg87nepswn2dj8"
LSMTokenBaseDenom = ValAddress + "/32"

DepositAddress = "deposit"
CommunityPoolStakeHoldingAddress = "staking-holding"
CommunityPoolRedeemHoldingAddress = "redeem-holding"
)

type KeeperTestSuite struct {
Expand Down Expand Up @@ -65,6 +60,16 @@ func (s *KeeperTestSuite) MustGetHostZone(chainId string) types.HostZone {
return hostZone
}

// Helper function to create an stride epoch tracker that dictates the timeout
func (s *KeeperTestSuite) CreateStrideEpochForICATimeout(timeoutDuration time.Duration) {
epochEndTime := uint64(s.Ctx.BlockTime().Add(timeoutDuration).UnixNano())
epochTracker := types.EpochTracker{
EpochIdentifier: epochtypes.STRIDE_EPOCH,
NextEpochStartTime: epochEndTime,
}
s.App.StakeibcKeeper.SetEpochTracker(s.Ctx, epochTracker)
}

func (s *KeeperTestSuite) TestIsRedemptionRateWithinSafetyBounds() {
params := s.App.StakeibcKeeper.GetParams(s.Ctx)
params.DefaultMinRedemptionRateThreshold = 75
Expand Down
112 changes: 71 additions & 41 deletions x/stakeibc/keeper/reward_converter.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,11 +25,11 @@ type PacketForwardMetadata struct {
Forward *ForwardMetadata `json:"forward"`
}
type ForwardMetadata struct {
Receiver string `json:"receiver,omitempty"`
Port string `json:"port,omitempty"`
Channel string `json:"channel,omitempty"`
Timeout string `json:"timeout,omitempty"`
Retries uint8 `json:"retries,omitempty"`
Receiver string `json:"receiver"`
Port string `json:"port"`
Channel string `json:"channel"`
Timeout string `json:"timeout"`
Retries int64 `json:"retries"`
}

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////
Expand All @@ -51,22 +51,17 @@ type ForwardMetadata struct {
// and the normal staking and distribution flow will continue from there.
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////

// ICA tx will kick off transfering the reward tokens from the hostZone withdrawl ICA to the tradeZone trade ICA
// This will be two hops to unwind the ibc denom through the rewardZone using pfm in the transfer memo if possible
//
// msgs with packet forwarding memos can unwind through the reward zone and chain two transfer hops without callbacks
func (k Keeper) TransferRewardTokensHostToTrade(ctx sdk.Context, amount sdkmath.Int, route types.TradeRoute) error {
// If the min swap amount was not set it would be ZeroInt, if positive we need to compare to the amount given
// then if the min swap amount is greater than the current amount, do nothing this epoch to avoid small transfers
// Particularly important for the PFM hop if the reward chain has frictional transfer fees (like noble chain)
if route.TradeConfig.MinSwapAmount.IsPositive() && route.TradeConfig.MinSwapAmount.GT(amount) {
return nil
}

// Builds a PFM transfer message to send reward tokens from the host zone,
// through the reward zone (to unwind) and finally to the trade zone
func (k Keeper) BuildHostToTradeTransferMsg(
ctx sdk.Context,
amount sdkmath.Int,
route types.TradeRoute,
) (msg transfertypes.MsgTransfer, err error) {
// Get the epoch tracker to determine the timeouts
strideEpochTracker, found := k.GetEpochTracker(ctx, epochstypes.STRIDE_EPOCH)
if !found {
return errorsmod.Wrapf(types.ErrEpochNotFound, epochstypes.STRIDE_EPOCH)
return msg, errorsmod.Wrapf(types.ErrEpochNotFound, epochstypes.STRIDE_EPOCH)
}

// Timeout the first transfer halfway through the epoch, and the second transfer at the end of the epoch
Expand Down Expand Up @@ -96,29 +91,48 @@ func (k Keeper) TransferRewardTokensHostToTrade(ctx sdk.Context, amount sdkmath.
}
memoJSON, err := json.Marshal(memo)
if err != nil {
return err
return msg, err
}

var msgs []proto.Message
msgs = append(msgs, &transfertypes.MsgTransfer{
msg = transfertypes.MsgTransfer{
SourcePort: transfertypes.PortID,
SourceChannel: route.HostToRewardChannelId, // channel on hostZone for transfers to rewardZone
Token: sendTokens,
Sender: withdrawlIcaAddress,
Receiver: unwindIcaAddress, // could be "pfm" or a real address depending on version
TimeoutTimestamp: transfer1TimeoutTimestamp,
Memo: string(memoJSON),
})
}

return msg, nil
}

// ICA tx will kick off transfering the reward tokens from the hostZone withdrawl ICA to the tradeZone trade ICA
// This will be two hops to unwind the ibc denom through the rewardZone using pfm in the transfer memo
func (k Keeper) TransferRewardTokensHostToTrade(ctx sdk.Context, amount sdkmath.Int, route types.TradeRoute) error {
// If the min swap amount was not set it would be ZeroInt, if positive we need to compare to the amount given
// then if the min swap amount is greater than the current amount, do nothing this epoch to avoid small transfers
// Particularly important for the PFM hop if the reward chain has frictional transfer fees (like noble chain)
if route.TradeConfig.MinSwapAmount.IsPositive() && route.TradeConfig.MinSwapAmount.GT(amount) {
return nil
}

// Build the PFM transfer message from host to trade zone
msg, err := k.BuildHostToTradeTransferMsg(ctx, amount, route)
if err != nil {
return err
}
msgs := []proto.Message{&msg}

hostZoneId := route.HostAccount.ChainId
rewardZoneId := route.RewardAccount.ChainId
tradeZoneId := route.TradeAccount.ChainId
k.Logger(ctx).Info(utils.LogWithHostZone(hostZoneId,
"Preparing MsgTransfer of %+v from %s to %s to %s", sendTokens, hostZoneId, rewardZoneId, tradeZoneId))
"Preparing MsgTransfer of %+v from %s to %s to %s", msg.Token, hostZoneId, rewardZoneId, tradeZoneId))

// Send the ICA tx to kick off transfer from hostZone through rewardZone to the tradeZone (no callbacks)
hostZoneConnectionId := route.HostAccount.ConnectionId
err = k.SubmitICATxWithoutCallback(ctx, hostZoneConnectionId, types.ICAAccountType_WITHDRAWAL, msgs, transfer1TimeoutTimestamp)
err = k.SubmitICATxWithoutCallback(ctx, hostZoneConnectionId, types.ICAAccountType_WITHDRAWAL, msgs, msg.TimeoutTimestamp)
if err != nil {
return errorsmod.Wrapf(err, "Failed to submit ICA tx, Messages: %+v", msgs)
}
Expand Down Expand Up @@ -167,19 +181,13 @@ func (k Keeper) TransferConvertedTokensTradeToHost(ctx sdk.Context, amount sdkma
return nil
}

// Trade reward tokens in the Trade ICA for the host denom tokens using ICA remote tx on trade zone
// The amount represents the total amount of the reward token in the trade ICA found by the calling ICQ
// Builds the Osmosis swap message to trade reward tokens for host tokens
// Depending on min and max swap amounts set in the route, it is possible not the full amount given will swap
func (k Keeper) SwapRewardTokens(ctx sdk.Context, rewardAmount sdkmath.Int, route types.TradeRoute) error {
// If the min swap amount was not set it would be ZeroInt, if positive we need to compare to the amount given
// then if the min swap amount is greater than the current amount, do nothing this epoch to avoid small swaps
tradeConfig := route.TradeConfig
if tradeConfig.MinSwapAmount.IsPositive() && tradeConfig.MinSwapAmount.GT(rewardAmount) {
return nil
}

// The minimum amount of tokens that can come out of the trade is calculated using a price from the pool
func (k Keeper) BuildSwapMsg(rewardAmount sdkmath.Int, route types.TradeRoute) (msg types.MsgSwapExactAmountIn, err error) {
// If the max swap amount was not set it would be ZeroInt, if positive we need to compare to the amount given
// then if max swap amount is LTE to amount full swap is possible so amount is fine, otherwise set amount to max
tradeConfig := route.TradeConfig
if tradeConfig.MaxSwapAmount.IsPositive() && rewardAmount.GT(tradeConfig.MaxSwapAmount) {
rewardAmount = tradeConfig.MaxSwapAmount
}
Expand All @@ -188,7 +196,7 @@ func (k Keeper) SwapRewardTokens(ctx sdk.Context, rewardAmount sdkmath.Int, rout
// The only time this should not be set is right after the pool is added,
// before an ICQ has been submitted for the price
if tradeConfig.SwapPrice.IsZero() {
return fmt.Errorf("Price not found for pool %d", tradeConfig.PoolId)
return msg, fmt.Errorf("Price not found for pool %d", tradeConfig.PoolId)
}

// If there is a valid price, use it to set a floor for the acceptable minimum output tokens
Expand All @@ -205,7 +213,6 @@ func (k Keeper) SwapRewardTokens(ctx sdk.Context, rewardAmount sdkmath.Int, rout
minOutPercentage := sdk.OneDec().Sub(tradeConfig.MaxAllowedSwapLossRate)
minOut := rewardAmountConverted.Mul(minOutPercentage).TruncateInt()

tradeIcaAccount := route.TradeAccount
tradeTokens := sdk.NewCoin(route.RewardDenomOnTradeZone, rewardAmount)

// Prepare Osmosis GAMM module MsgSwapExactAmountIn from the trade account to perform the trade
Expand All @@ -215,23 +222,46 @@ func (k Keeper) SwapRewardTokens(ctx sdk.Context, rewardAmount sdkmath.Int, rout
PoolId: tradeConfig.PoolId,
TokenOutDenom: route.HostDenomOnTradeZone,
}}
msgs := []proto.Message{&types.MsgSwapExactAmountIn{
Sender: tradeIcaAccount.Address,
msg = types.MsgSwapExactAmountIn{
Sender: route.TradeAccount.Address,
Routes: routes,
TokenIn: tradeTokens,
TokenOutMinAmount: minOut,
}}
}

return msg, nil
}

// Trade reward tokens in the Trade ICA for the host denom tokens using ICA remote tx on trade zone
// The amount represents the total amount of the reward token in the trade ICA found by the calling ICQ
func (k Keeper) SwapRewardTokens(ctx sdk.Context, rewardAmount sdkmath.Int, route types.TradeRoute) error {
// If the min swap amount was not set it would be ZeroInt, if positive we need to compare to the amount given
// then if the min swap amount is greater than the current amount, do nothing this epoch to avoid small swaps
tradeConfig := route.TradeConfig
if tradeConfig.MinSwapAmount.IsPositive() && tradeConfig.MinSwapAmount.GT(rewardAmount) {
return nil
}

// Build the Osmosis swap message to convert reward tokens to host tokens
msg, err := k.BuildSwapMsg(rewardAmount, route)
if err != nil {
return err
}
msgs := []proto.Message{&msg}

tradeIcaAccount := route.TradeAccount
k.Logger(ctx).Info(utils.LogWithHostZone(tradeIcaAccount.ChainId,
"Preparing MsgSwapExactAmountIn of %+v from the trade account", tradeTokens))
"Preparing MsgSwapExactAmountIn of %+v from the trade account", msg.TokenIn))

// Timeout the swap at the end of the epoch
strideEpochTracker, found := k.GetEpochTracker(ctx, epochstypes.STRIDE_EPOCH)
if !found {
return errorsmod.Wrapf(types.ErrEpochNotFound, epochstypes.STRIDE_EPOCH)
}
timeout := uint64(strideEpochTracker.NextEpochStartTime)

// Send the ICA tx to perform the swap on the tradeZone
err := k.SubmitICATxWithoutCallback(ctx, tradeIcaAccount.ConnectionId, types.ICAAccountType_CONVERTER_TRADE, msgs, timeout)
err = k.SubmitICATxWithoutCallback(ctx, tradeIcaAccount.ConnectionId, types.ICAAccountType_CONVERTER_TRADE, msgs, timeout)
if err != nil {
return errorsmod.Wrapf(err, "Failed to submit ICA tx for the swap, Messages: %v", msgs)
}
Expand Down
Loading

0 comments on commit fe21ad3

Please sign in to comment.