forked from Canto-Network/Canto
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathutils.go
111 lines (92 loc) · 4.15 KB
/
utils.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
package ibc
import (
errorsmod "cosmossdk.io/errors"
sdkmath "cosmossdk.io/math"
sdk "github.com/cosmos/cosmos-sdk/types"
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
transfertypes "github.com/cosmos/ibc-go/v8/modules/apps/transfer/types"
channeltypes "github.com/cosmos/ibc-go/v8/modules/core/04-channel/types"
canto "github.com/Canto-Network/Canto/v7/types"
)
// GetTransferSenderRecipient returns the sender and recipient sdk.AccAddresses
// from an ICS20 FungibleTokenPacketData as well as the original sender bech32
// address from the packet data. This function fails if:
// - the packet data is not FungibleTokenPacketData
// - sender address is invalid
// - recipient address is invalid
func GetTransferSenderRecipient(packet channeltypes.Packet) (
sender, recipient sdk.AccAddress,
senderBech32, recipientBech32 string,
err error,
) {
// unmarshal packet data to obtain the sender and recipient
var data transfertypes.FungibleTokenPacketData
if err := transfertypes.ModuleCdc.UnmarshalJSON(packet.GetData(), &data); err != nil {
return nil, nil, "", "", errorsmod.Wrapf(sdkerrors.ErrUnknownRequest, "cannot unmarshal ICS-20 transfer packet data")
}
// validate the sender bech32 address from the counterparty chain
// and change the bech32 human readable prefix (HRP) of the sender to `canto`
sender, err = canto.GetcantoAddressFromBech32(data.Sender)
if err != nil {
return nil, nil, "", "", errorsmod.Wrap(err, "invalid sender")
}
// validate the recipient bech32 address from the counterparty chain
// and change the bech32 human readable prefix (HRP) of the recipient to `canto`
recipient, err = canto.GetcantoAddressFromBech32(data.Receiver)
if err != nil {
return nil, nil, "", "", errorsmod.Wrap(err, "invalid recipient")
}
return sender, recipient, data.Sender, data.Receiver, nil
}
// GetTransferAmount returns the amount from an ICS20 FungibleTokenPacketData.
func GetTransferAmount(packet channeltypes.Packet) (string, error) {
// unmarshal packet data to obtain the sender and recipient
var data transfertypes.FungibleTokenPacketData
if err := transfertypes.ModuleCdc.UnmarshalJSON(packet.GetData(), &data); err != nil {
return "", errorsmod.Wrapf(sdkerrors.ErrUnknownRequest, "cannot unmarshal ICS-20 transfer packet data")
}
if data.Amount == "" {
return "", errorsmod.Wrapf(sdkerrors.ErrInvalidCoins, "empty amount")
}
if _, ok := sdkmath.NewIntFromString(data.Amount); !ok {
return "", errorsmod.Wrapf(sdkerrors.ErrInvalidCoins, "invalid amount")
}
return data.Amount, nil
}
// GetReceivedCoin returns the transferred coin from an ICS20 FungibleTokenPacketData
// as seen from the destination chain.
// If the receiving chain is the source chain of the tokens, it removes the prefix
// path added by source (i.e sender) chain to the denom. Otherwise, it adds the
// prefix path from the destination chain to the denom.
func GetReceivedCoin(srcPort, srcChannel, dstPort, dstChannel, rawDenom, rawAmt string) sdk.Coin {
// NOTE: Denom and amount are already validated
amount, _ := sdkmath.NewIntFromString(rawAmt)
if transfertypes.ReceiverChainIsSource(srcPort, srcChannel, rawDenom) {
// remove prefix added by sender chain
voucherPrefix := transfertypes.GetDenomPrefix(srcPort, srcChannel)
unprefixedDenom := rawDenom[len(voucherPrefix):]
// coin denomination used in sending from the escrow address
denom := unprefixedDenom
// The denomination used to send the coins is either the native denom or the hash of the path
// if the denomination is not native.
denomTrace := transfertypes.ParseDenomTrace(unprefixedDenom)
if denomTrace.Path != "" {
denom = denomTrace.IBCDenom()
}
return sdk.Coin{
Denom: denom,
Amount: amount,
}
}
// since SendPacket did not prefix the denomination, we must prefix denomination here
sourcePrefix := transfertypes.GetDenomPrefix(dstPort, dstChannel)
// NOTE: sourcePrefix contains the trailing "/"
prefixedDenom := sourcePrefix + rawDenom
// construct the denomination trace from the full raw denomination
denomTrace := transfertypes.ParseDenomTrace(prefixedDenom)
voucherDenom := denomTrace.IBCDenom()
return sdk.Coin{
Denom: voucherDenom,
Amount: amount,
}
}