Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

autopilot fallback address #1039

Merged
merged 34 commits into from
Jan 11, 2024
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
d3e0501
pull in LS + forward changes from https://github.com/Stride-Labs/stri…
asalzmann Nov 28, 2023
1c6ed0b
integration test working
asalzmann Nov 28, 2023
596250b
fix unittest
asalzmann Nov 28, 2023
220c8f1
restrict to either autopilot or pfm
sampocs Dec 21, 2023
528994a
added hash receiver helper
sampocs Dec 22, 2023
264cc28
added bank keeper to autopilot
sampocs Dec 22, 2023
45b1465
first pass at hashed recipient implementation
sampocs Dec 22, 2023
99db724
cleaned up variable names
sampocs Dec 22, 2023
f9d5d09
cleanup and docs
sampocs Dec 22, 2023
3bdfcd7
moved types to autopilot
sampocs Dec 22, 2023
eb408d2
walked back almost all the changes from above :(
sampocs Dec 22, 2023
8ee8ef2
cleanup again
sampocs Dec 22, 2023
b1d4503
added bank keeper again
sampocs Dec 27, 2023
208eac7
added bank send to hashed sender
sampocs Dec 27, 2023
2c22d17
renamed hashed address function
sampocs Dec 28, 2023
c0ca4f5
added keepers to store fallback address
sampocs Dec 28, 2023
a119b8d
implemented onAck and onTimeout
sampocs Dec 28, 2023
f102574
added unit test for helpers
sampocs Dec 28, 2023
619c480
added unit tests for on ack packet
sampocs Dec 29, 2023
baaeb50
added unit tests for full callbacks
sampocs Dec 29, 2023
c8288dc
info -> error log
sampocs Dec 29, 2023
299f585
autopilot liquid stake and forward unit tests (#1041)
sampocs Jan 4, 2024
1bfa7ca
renamed key function
sampocs Jan 9, 2024
cd7ce0b
replaced CheckAckStatus with helper from icacallbacks
sampocs Jan 9, 2024
8958d80
removed retry and replaced with send to fallback address
sampocs Jan 9, 2024
55635d0
fixed unit tests
sampocs Jan 9, 2024
2bf291f
updated timeout to 3 hours
sampocs Jan 9, 2024
bd61e0a
Merge branch 'main' into sam/autopilot-hash-sender-2
sampocs Jan 9, 2024
a814f92
updated timeout comments
sampocs Jan 9, 2024
5042669
updated checkmes
sampocs Jan 9, 2024
158c77d
Merge branch 'sam/autopilot-hash-sender-2' into sam/autopilot-fallbac…
sampocs Jan 9, 2024
158ccec
Merge branch 'main' into sam/autopilot-fallback-address
sampocs Jan 10, 2024
ba9330c
fixed unit test after merge
sampocs Jan 11, 2024
2bec06a
Merge branch 'main' into sam/autopilot-fallback-address
sampocs Jan 11, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion app/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -612,8 +612,11 @@ func NewStrideApp(
appCodec,
keys[autopilottypes.StoreKey],
app.GetSubspace(autopilottypes.ModuleName),
app.BankKeeper,
app.StakeibcKeeper,
app.ClaimKeeper)
app.ClaimKeeper,
app.TransferKeeper,
)
autopilotModule := autopilot.NewAppModule(appCodec, app.AutopilotKeeper)

app.VestingKeeper = evmosvestingkeeper.NewKeeper(
Expand Down
7 changes: 7 additions & 0 deletions dockernet/config.sh
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,13 @@ STWALK_DENOM="stuwalk"
STEVMOS_DENOM="staevmos"
STDYDX_DENOM="studydx"

IBC_GAIA_CHANNEL_0_STATOM_DENOM='ibc/054A44EC8D9B68B9A6F0D5708375E00A5569A28F21E0064FF12CADC3FEF1D04F'
IBC_GAIA_CHANNEL_1_STATOM_DENOM='ibc/8B21DA0E34A49AE151FEEBCCF3AFE1188E24BA8E19439FB93434DF6008E7E228'
IBC_GAIA_CHANNEL_2_STATOM_DENOM='ibc/60CB7A5465C318C8F68F603D78721A2ECC1DA2D0E905C6AD9ACD1CAC3F0DB22D'
IBC_GAIA_CHANNEL_3_STATOM_DENOM='ibc/0C0FD07C29EB075C18EA77B73CF9FCE68A268E0738C9F5B11D13E418AD889437'

IBC_GAIA_STATOM_DENOM=$IBC_GAIA_CHANNEL_0_STATOM_DENOM

IBC_STRD_DENOM='ibc/FF6C2E86490C1C4FBBD24F55032831D2415B9D7882F85C3CC9C2401D79362BEA'

IBC_GAIA_CHANNEL_0_DENOM='ibc/27394FB092D2ECCD56123C74F36E4C1F926001CEADA9CA97EA622B25F41E5EB2'
Expand Down
28 changes: 28 additions & 0 deletions dockernet/tests/integration_tests.bats
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,34 @@ setup_file() {
assert_equal "$diff" $STAKE_AMOUNT
}

@test "[INTEGRATION-BASIC-$CHAIN_NAME] packet forwarding automatically liquid stake and ibc transfer stAsset to original network" {
memo='{ "autopilot": { "receiver": "'"$(STRIDE_ADDRESS)"'", "stakeibc": { "action": "LiquidStake", "ibc_receiver": "'$HOST_VAL_ADDRESS'" } } }'

# get initial balances
stibctoken_balance_start=$($HOST_MAIN_CMD q bank balances $HOST_VAL_ADDRESS --denom $IBC_GAIA_STATOM_DENOM | GETBAL)

# Send the IBC transfer with the JSON memo
transfer_msg_prefix="$HOST_MAIN_CMD tx ibc-transfer transfer transfer $HOST_TRANSFER_CHANNEL"
if [[ "$CHAIN_NAME" == "GAIA" ]]; then
# For GAIA (ibc-v3), pass the memo into the receiver field
$transfer_msg_prefix "$memo" ${PACKET_FORWARD_STAKE_AMOUNT}${HOST_DENOM} --from $HOST_VAL -y
elif [[ "$CHAIN_NAME" == "HOST" ]]; then
# For HOST (ibc-v5), pass an address for a receiver and the memo in the --memo field
$transfer_msg_prefix $(STRIDE_ADDRESS) ${PACKET_FORWARD_STAKE_AMOUNT}${HOST_DENOM} --memo "$memo" --from $HOST_VAL -y
else
# For all other hosts, skip this test
skip "Packet forward liquid stake test is only run on GAIA and HOST"
fi

# Wait for the transfer to complete
WAIT_FOR_BALANCE_CHANGE $CHAIN_NAME $HOST_VAL_ADDRESS $IBC_GAIA_STATOM_DENOM

# make sure stATOM balance increased
stibctoken_balance_end=$($HOST_MAIN_CMD q bank balances $HOST_VAL_ADDRESS --denom $IBC_GAIA_STATOM_DENOM | GETBAL)
stibctoken_balance_diff=$(($stibctoken_balance_end-$stibctoken_balance_start))
assert_equal "$stibctoken_balance_diff" "$PACKET_FORWARD_STAKE_AMOUNT"
}

# check that tokens on the host are staked
@test "[INTEGRATION-BASIC-$CHAIN_NAME] tokens on $CHAIN_NAME were staked" {
# wait for another epoch to pass so that tokens are staked
Expand Down
13 changes: 6 additions & 7 deletions x/autopilot/keeper/airdrop.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,16 +12,15 @@ import (
channeltypes "github.com/cosmos/ibc-go/v7/modules/core/04-channel/types"

"github.com/Stride-Labs/stride/v16/utils"
"github.com/Stride-Labs/stride/v16/x/autopilot/types"
claimtypes "github.com/Stride-Labs/stride/v16/x/claim/types"
stakeibctypes "github.com/Stride-Labs/stride/v16/x/stakeibc/types"
)

// Attempt to link a host address with a stride address to enable airdrop claims
func (k Keeper) TryUpdateAirdropClaim(
ctx sdk.Context,
packet channeltypes.Packet,
data transfertypes.FungibleTokenPacketData,
packetMetadata types.ClaimPacketMetadata,
transferMetadata transfertypes.FungibleTokenPacketData,
) error {
params := k.GetParams(ctx)
if !params.ClaimActive {
Expand All @@ -39,11 +38,11 @@ func (k Keeper) TryUpdateAirdropClaim(
}

// grab relevant addresses
senderStrideAddress := utils.ConvertAddressToStrideAddress(data.Sender)
senderStrideAddress := utils.ConvertAddressToStrideAddress(transferMetadata.Sender)
if senderStrideAddress == "" {
return errorsmod.Wrapf(sdkerrors.ErrInvalidAddress, fmt.Sprintf("invalid sender address (%s)", data.Sender))
return errorsmod.Wrapf(sdkerrors.ErrInvalidAddress, fmt.Sprintf("invalid sender address (%s)", transferMetadata.Sender))
}
newStrideAddress := packetMetadata.StrideAddress
newStrideAddress := transferMetadata.Receiver

// find the airdrop for this host chain ID
airdrop, found := k.claimKeeper.GetAirdropByChainId(ctx, hostZone.ChainId)
Expand All @@ -56,7 +55,7 @@ func (k Keeper) TryUpdateAirdropClaim(

airdropId := airdrop.AirdropIdentifier
k.Logger(ctx).Info(fmt.Sprintf("updating airdrop address %s (orig %s) to %s for airdrop %s",
senderStrideAddress, data.Sender, newStrideAddress, airdropId))
senderStrideAddress, transferMetadata.Sender, newStrideAddress, airdropId))

return k.claimKeeper.UpdateAirdropAddress(ctx, senderStrideAddress, newStrideAddress, airdropId)
}
2 changes: 1 addition & 1 deletion x/autopilot/keeper/airdrop_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -245,7 +245,7 @@ func (s *KeeperTestSuite) TestAirdropOnRecvPacket() {
destinationPortID: transfertypes.PortID,
packetData: transfertypes.FungibleTokenPacketData{
Receiver: strideAddress,
Memo: strings.Repeat("X", 300),
Memo: strings.Repeat("X", 513),
},
transferShouldSucceed: false,
airdropShouldUpdate: false,
Expand Down
39 changes: 39 additions & 0 deletions x/autopilot/keeper/fallback.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
package keeper

import (
"github.com/cosmos/cosmos-sdk/store/prefix"
sdk "github.com/cosmos/cosmos-sdk/types"

"github.com/Stride-Labs/stride/v16/x/autopilot/types"
)

// Stores a fallback address for an outbound transfer
func (k Keeper) SetTransferFallbackAddress(ctx sdk.Context, channelId string, sequence uint64, address string) {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
store := prefix.NewStore(ctx.KVStore(k.storeKey), types.TransferFallbackAddressPrefix)
riley-stride marked this conversation as resolved.
Show resolved Hide resolved
key := types.GetTransferFallbackAddressKeyPrefix(channelId, sequence)
value := []byte(address)
store.Set(key, value)
}

// Removes a fallback address from the store
// This is used after the ack or timeout for a packet has been received
func (k Keeper) RemoveTransferFallbackAddress(ctx sdk.Context, channelId string, sequence uint64) {
store := prefix.NewStore(ctx.KVStore(k.storeKey), types.TransferFallbackAddressPrefix)
key := types.GetTransferFallbackAddressKeyPrefix(channelId, sequence)
sampocs marked this conversation as resolved.
Show resolved Hide resolved
store.Delete(key)
}

// Returns a fallback address, given the channel ID and sequence number of the packet
// If no fallback address has been stored, return false
func (k Keeper) GetTransferFallbackAddress(ctx sdk.Context, channelId string, sequence uint64) (address string, found bool) {
store := prefix.NewStore(ctx.KVStore(k.storeKey), types.TransferFallbackAddressPrefix)

key := types.GetTransferFallbackAddressKeyPrefix(channelId, sequence)
valueBz := store.Get(key)

if len(valueBz) == 0 {
return "", false
}

return string(valueBz), true
}
22 changes: 22 additions & 0 deletions x/autopilot/keeper/fallback_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package keeper_test

// Tests Get/Set/RemoveTransferFallbackAddress
func (s *KeeperTestSuite) TestTransferFallbackAddress() {
channelId := "channel-0"
sequence := uint64(100)
expectedAddress := "stride1xjp08gxef09fck6yj2lg0vrgpcjhqhp055ffhj"

// Add a new fallback address
s.App.AutopilotKeeper.SetTransferFallbackAddress(s.Ctx, channelId, sequence, expectedAddress)

// Confirm we can retrieve it
actualAddress, found := s.App.AutopilotKeeper.GetTransferFallbackAddress(s.Ctx, channelId, sequence)
s.Require().True(found, "address should have been found")
s.Require().Equal(expectedAddress, actualAddress, "fallback addres")

// Remove it and confirm we can no longer retrieve it
s.App.AutopilotKeeper.RemoveTransferFallbackAddress(s.Ctx, channelId, sequence)

_, found = s.App.AutopilotKeeper.GetTransferFallbackAddress(s.Ctx, channelId, sequence)
s.Require().False(found, "address should have been removed")
}
162 changes: 162 additions & 0 deletions x/autopilot/keeper/ibc.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,162 @@
package keeper

import (
"fmt"

errorsmod "cosmossdk.io/errors"
sdkmath "cosmossdk.io/math"
sdk "github.com/cosmos/cosmos-sdk/types"
transfertypes "github.com/cosmos/ibc-go/v7/modules/apps/transfer/types"
channeltypes "github.com/cosmos/ibc-go/v7/modules/core/04-channel/types"
)

// Deserializes the acknowledgement and returns a bool indicating whether it was successful or was an ack error
func (k Keeper) CheckAcknowledgementStatus(ctx sdk.Context, acknowledgementBz []byte) (success bool, err error) {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
// Unmarshal the raw ack response
var acknowledgement channeltypes.Acknowledgement
if err := transfertypes.ModuleCdc.UnmarshalJSON(acknowledgementBz, &acknowledgement); err != nil {
return false, errorsmod.Wrapf(err, "cannot unmarshal ICS-20 transfer packet acknowledgement")
}

// The ack can come back as either AcknowledgementResult (success) or AcknowledgementError (failure)
switch response := acknowledgement.Response.(type) {
case *channeltypes.Acknowledgement_Result:
if len(response.Result) == 0 {
return false, errorsmod.Wrapf(channeltypes.ErrInvalidAcknowledgement, "acknowledgement result cannot be empty")
}
return true, nil

case *channeltypes.Acknowledgement_Error:
k.Logger(ctx).Error(fmt.Sprintf("autopilot acknowledgement error: %s", response.Error))
return false, nil

default:
return false, errorsmod.Wrapf(channeltypes.ErrInvalidAcknowledgement, "unsupported acknowledgement response field type %T", response)
}
}

// Build an sdk.Coin type from the transfer metadata which includes strings for the amount and denom
func (k Keeper) BuildCoinFromTransferMetadata(transferMetadata transfertypes.FungibleTokenPacketData) (coin sdk.Coin, err error) {
amount, ok := sdkmath.NewIntFromString(transferMetadata.Amount)
if !ok {
return coin, fmt.Errorf("unable to parse amount from transfer packet: %v", transferMetadata)
}
coin = sdk.NewCoin(transferMetadata.Denom, amount)
return coin, nil
}

// In the event of an ack error after a outbound transfer, we'll have to bank send to a fallback address
func (k Keeper) SendToFallbackAddress(ctx sdk.Context, packetData []byte, fallbackAddress string) error {
// First unmarshal the transfer metadata to get the sender/reciever, and token amount/denom
var transferMetadata transfertypes.FungibleTokenPacketData
if err := transfertypes.ModuleCdc.UnmarshalJSON(packetData, &transferMetadata); err != nil {
return err
}

// Pull out the original sender of the transfer which will also be the bank sender
sender := transferMetadata.Sender
senderAccount, err := sdk.AccAddressFromBech32(sender)
if err != nil {
return errorsmod.Wrapf(err, "invalid sender address")
}
fallbackAccount, err := sdk.AccAddressFromBech32(fallbackAddress)
if err != nil {
return errorsmod.Wrapf(err, "invalid fallback address")
}

// Build the token from the transfer metadata
token, err := k.BuildCoinFromTransferMetadata(transferMetadata)
if err != nil {
return err
}

// Finally send to the fallback account
if err := k.bankKeeper.SendCoins(ctx, senderAccount, fallbackAccount, sdk.NewCoins(token)); err != nil {
return err
}

return nil
}

// If there was a failed ack from an outbound transfer of one of the autopilot actions,
// we'll need to check if there was a fallback address. If one was stored, bank send
// to that fallback address
// If the ack was successful, we should delete the address (if it exists)
func (k Keeper) OnAcknowledgementPacket(ctx sdk.Context, packet channeltypes.Packet, acknowledgement []byte) error {
// Retrieve the fallback address for the given packet
// We use the packet source channel here since this will correspond with the channel on Stride
channelId := packet.SourceChannel
sequence := packet.Sequence
fallbackAddress, fallbackAddressFound := k.GetTransferFallbackAddress(ctx, channelId, sequence)

// If there was no fallback address, there's nothing else to do
if !fallbackAddressFound {
riley-stride marked this conversation as resolved.
Show resolved Hide resolved
return nil
}

// Remove the fallback address since the packet is no longer pending
k.RemoveTransferFallbackAddress(ctx, channelId, sequence)

// Check whether the ack was successful or was an ack error
success, err := k.CheckAcknowledgementStatus(ctx, acknowledgement)
if err != nil {
return err
}

// If successful, no additional action is necessary
if success {
return nil
}

// If the ack was an error, we'll need to bank send to the fallback address
return k.SendToFallbackAddress(ctx, packet.Data, fallbackAddress)
}

// If there's a timed out packet, we'll infinitely retry the transfer
func (k Keeper) OnTimeoutPacket(ctx sdk.Context, packet channeltypes.Packet) error {
sampocs marked this conversation as resolved.
Show resolved Hide resolved
// Retrieve the fallback address from the original packet
// We use the packet source channel here since this will correspond with the channel on Stride
channelId := packet.SourceChannel
originalSequence := packet.Sequence
fallbackAddress, fallbackAddressFound := k.GetTransferFallbackAddress(ctx, channelId, originalSequence)

// If there was no fallback address, this packet was not from an autopilot action and there's no need to retry
if !fallbackAddressFound {
return nil
}

// If this was from an autopilot action, unmarshal the transfer metadata to get the original transfer info
var transferMetadata transfertypes.FungibleTokenPacketData
if err := transfertypes.ModuleCdc.UnmarshalJSON(packet.Data, &transferMetadata); err != nil {
return errorsmod.Wrapf(err, "unable to unmarshal ICS-20 packet data")
}

// Build the token from the transfer metadata
token, err := k.BuildCoinFromTransferMetadata(transferMetadata)
if err != nil {
return err
}

// Submit the transfer again with a new timeout
timeoutTimestamp := uint64(ctx.BlockTime().UnixNano()) + transfertypes.DefaultRelativePacketTimeoutTimestamp
msgTransfer := transfertypes.MsgTransfer{
SourcePort: transfertypes.PortID,
SourceChannel: packet.SourceChannel,
Token: token,
Sender: transferMetadata.Sender,
Receiver: transferMetadata.Receiver,
TimeoutTimestamp: timeoutTimestamp,
Memo: transferMetadata.Memo,
}
retryResponse, err := k.transferKeeper.Transfer(ctx, &msgTransfer)
if err != nil {
return errorsmod.Wrapf(err, "unable to submit transfer retry of %+v", msgTransfer)
}

// Update the fallback address to use the new sequence number
updatedSequence := retryResponse.Sequence
k.RemoveTransferFallbackAddress(ctx, channelId, originalSequence)
k.SetTransferFallbackAddress(ctx, channelId, updatedSequence, fallbackAddress)

return nil
}
Loading