/
msg_server_take_multi_asset_deposit.go
116 lines (93 loc) · 3.38 KB
/
msg_server_take_multi_asset_deposit.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
package keeper
import (
"context"
"strings"
sdk "github.com/cosmos/cosmos-sdk/types"
errorsmod "github.com/cosmos/cosmos-sdk/types/errors"
"github.com/sideprotocol/ibcswap/v6/modules/apps/101-interchain-swap/types"
)
func (k Keeper) TakeMultiAssetDeposit(ctx context.Context, msg *types.MsgTakeMultiAssetDepositRequest) (*types.MsgMultiAssetDepositResponse, error) {
sdkCtx := sdk.UnwrapSDKContext(ctx)
// Validate message
err := msg.ValidateBasic()
if err != nil {
return nil, err
}
pool, found := k.GetInterchainLiquidityPool(sdkCtx, msg.PoolId)
if !found {
return nil, errorsmod.Wrapf(types.ErrFailedMultiAssetDeposit, "%s", types.ErrNotFoundPool)
}
order, found := k.GetMultiDepositOrder(sdkCtx, msg.PoolId, msg.OrderId)
if !found {
return nil, errorsmod.Wrapf(types.ErrNotFoundMultiDepositOrder, "%s", types.ErrFailedMultiAssetDeposit)
}
if msg.Sender != order.DestinationTaker {
return nil, errorsmod.Wrapf(types.ErrMultipleAssetDepositNotAllowed, "due to %s of other's", types.ErrFailedMultiAssetDeposit)
}
if order.Status == types.OrderStatus_COMPLETE {
return nil, errorsmod.Wrapf(types.ErrAlreadyCompletedOrder, "due to %s of other's", types.ErrFailedMultiAssetDeposit)
}
// estimate pool token
amm := types.NewInterchainMarketMaker(&pool)
poolTokens, err := amm.DepositMultiAsset(sdk.Coins{
*order.Deposits[0],
*order.Deposits[1],
})
// check asset owned status
asset := order.Deposits[1]
if err != nil {
return nil, errorsmod.Wrapf(err, "due to %s of other's", types.ErrFailedMultiAssetDeposit)
}
balance := k.bankKeeper.GetBalance(sdkCtx, sdk.MustAccAddressFromBech32(msg.Sender), asset.Denom)
if balance.Amount.LT(asset.Amount) {
return nil, errorsmod.Wrapf(types.ErrInEnoughAmount, "due to %s of Lp", types.ErrFailedMultiAssetDeposit)
}
// Create escrow module account here
err = k.LockTokens(sdkCtx, pool.CounterPartyPort, pool.CounterPartyChannel, sdk.MustAccAddressFromBech32(msg.Sender), sdk.NewCoins(*asset))
if err != nil {
return nil, errorsmod.Wrapf(err, "due to %s", types.ErrFailedMultiAssetDeposit)
}
// Construct IBC packet
rawMsgData := types.ModuleCdc.MustMarshalJSON(msg)
rawStateChange := types.ModuleCdc.MustMarshalJSON(&types.StateChange{PoolTokens: poolTokens})
packet := types.IBCSwapPacketData{
Type: types.TAKE_MULTI_DEPOSIT,
Data: rawMsgData,
StateChange: rawStateChange,
}
timeoutHeight, timeoutStamp := types.GetDefaultTimeOut(&sdkCtx)
// Use input timeoutHeight, timeoutStamp
if msg.TimeoutHeight != nil {
timeoutHeight = *msg.TimeoutHeight
}
if msg.TimeoutTimeStamp != 0 {
timeoutStamp = msg.TimeoutTimeStamp
}
_, err = k.SendIBCSwapPacket(sdkCtx, msg.Port, msg.Channel, timeoutHeight, timeoutStamp, packet)
if err != nil {
return nil, err
}
// Emit events
poolTokenEventValues := []string{}
for _, poolToken := range poolTokens {
poolTokenEventValues = append(poolTokenEventValues, poolToken.String())
}
mintedPoolTokens := strings.Join(poolTokenEventValues, ":")
// emit events
k.EmitEvent(
sdkCtx, types.EventValueActionTakeMultiDeposit, msg.PoolId, msg.Sender,
sdk.Attribute{
Key: types.AttributeKeyPoolCreator,
Value: msg.Sender,
},
sdk.Attribute{
Key: types.AttributeKeyMultiDepositOrderId,
Value: msg.OrderId,
},
sdk.Attribute{
Key: types.AttributeKeyLpToken,
Value: mintedPoolTokens,
},
)
return &types.MsgMultiAssetDepositResponse{}, nil
}