Skip to content

Commit dcd0681

Browse files
refactor: channel handshake version improvements (#1283)
* refactor: returning version from OnChanOpenInit * refactor: update tests and add version to proto resp * refactor: adding version to msg server resp * refactor: remove unncessary if & update version on Endpoint.Ack * fix: ics29 OnChanOpenInit remake versionMetaData before returning * chore: update godoc * test: adding check for expected version string * test: adding test case for passing empty version string to ics20 onChanOpenInit * Update modules/apps/29-fee/ibc_module.go Co-authored-by: colin axnér <25233464+colin-axner@users.noreply.github.com> * chore: comment * chore: changelog * fix: ica now discards auth module version * chore: update changelog * adding default version for ics29 * fix: using transfer module directly rather than calling full middleware stack * fix testing bug * refactor: test now uses bool for isFeeEnabled rather than direct check * docs: updating comment and migration docs Co-authored-by: colin axnér <25233464+colin-axner@users.noreply.github.com>
1 parent ce7ac2e commit dcd0681

File tree

15 files changed

+151
-57
lines changed

15 files changed

+151
-57
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ Ref: https://keepachangelog.com/en/1.0.0/
4343
### API Breaking
4444

4545
* (transfer) [\#1250](https://github.com/cosmos/ibc-go/pull/1250) Deprecate `GetTransferAccount` since the `transfer` module account is never used.
46+
* (channel) [\#1283](https://github.com/cosmos/ibc-go/pull/1283) The `OnChanOpenInit` application callback now returns a version string in line with the latest [spec changes](https://github.com/cosmos/ibc/pull/629).
4647
* (modules/29-fee)[\#1338](https://github.com/cosmos/ibc-go/pull/1338) Renaming `Result` field in `IncentivizedAcknowledgement` to `AppAcknowledgement`.
4748
* (modules/29-fee)[\#1343](https://github.com/cosmos/ibc-go/pull/1343) Renaming `KeyForwardRelayerAddress` to `KeyRelayerAddressForAsyncAck`, and `ParseKeyForwardRelayerAddress` to `ParseKeyRelayerAddressForAsyncAck`.
4849

docs/migrations/v3-to-v4.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
# Migrating from ibc-go v2 to v3
1+
# Migrating from ibc-go v3 to v4
22

33
This document is intended to highlight significant changes which may require more information than presented in the CHANGELOG.
44
Any changes that must be done by a user of ibc-go should be documented here.
@@ -23,4 +23,8 @@ No genesis or in-place migrations required when upgrading from v1 or v2 of ibc-g
2323
The `WriteAcknowledgement` API now takes the `exported.Acknowledgement` type instead of passing in the acknowledgement byte array directly.
2424
This is an API breaking change and as such IBC application developers will have to update any calls to `WriteAcknowledgement`.
2525

26+
The `OnChanOpenInit` application callback has been modified.
27+
The return signature now includes the application version as detailed in the latest IBC [spec changes](https://github.com/cosmos/ibc/pull/629).
28+
29+
2630

modules/apps/27-interchain-accounts/controller/ibc_middleware.go

+11-6
Original file line numberDiff line numberDiff line change
@@ -45,18 +45,23 @@ func (im IBCMiddleware) OnChanOpenInit(
4545
chanCap *capabilitytypes.Capability,
4646
counterparty channeltypes.Counterparty,
4747
version string,
48-
) error {
48+
) (string, error) {
4949
if !im.keeper.IsControllerEnabled(ctx) {
50-
return types.ErrControllerSubModuleDisabled
50+
return "", types.ErrControllerSubModuleDisabled
5151
}
5252

5353
if err := im.keeper.OnChanOpenInit(ctx, order, connectionHops, portID, channelID, chanCap, counterparty, version); err != nil {
54-
return err
54+
return "", err
55+
}
56+
57+
// call underlying app's OnChanOpenInit callback with the passed in version
58+
// the version returned is discarded as the ica-auth module does not have permission to edit the version string.
59+
// ics27 will always return the version string containing the Metadata struct which is created during the `RegisterInterchainAccount` call.
60+
if _, err := im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID, chanCap, counterparty, version); err != nil {
61+
return "", err
5562
}
5663

57-
// call underlying app's OnChanOpenInit callback with the appVersion
58-
return im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID,
59-
chanCap, counterparty, version)
64+
return version, nil
6065
}
6166

6267
// OnChanOpenTry implements the IBCMiddleware interface

modules/apps/27-interchain-accounts/controller/ibc_middleware_test.go

+16-3
Original file line numberDiff line numberDiff line change
@@ -147,8 +147,8 @@ func (suite *InterchainAccountsTestSuite) TestOnChanOpenInit() {
147147
suite.chainA.GetSimApp().ICAAuthModule.IBCApp.OnChanOpenInit = func(ctx sdk.Context, order channeltypes.Order, connectionHops []string,
148148
portID, channelID string, chanCap *capabilitytypes.Capability,
149149
counterparty channeltypes.Counterparty, version string,
150-
) error {
151-
return fmt.Errorf("mock ica auth fails")
150+
) (string, error) {
151+
return "", fmt.Errorf("mock ica auth fails")
152152
}
153153
}, false,
154154
},
@@ -197,11 +197,24 @@ func (suite *InterchainAccountsTestSuite) TestOnChanOpenInit() {
197197
cbs, ok := suite.chainA.App.GetIBCKeeper().Router.GetRoute(module)
198198
suite.Require().True(ok)
199199

200-
err = cbs.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
200+
version, err := cbs.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
201201
path.EndpointA.ChannelConfig.PortID, path.EndpointA.ChannelID, chanCap, channel.Counterparty, channel.GetVersion(),
202202
)
203203

204204
if tc.expPass {
205+
expMetadata := icatypes.NewMetadata(
206+
icatypes.Version,
207+
path.EndpointA.ConnectionID,
208+
path.EndpointB.ConnectionID,
209+
"",
210+
icatypes.EncodingProtobuf,
211+
icatypes.TxTypeSDKMultiMsg,
212+
)
213+
214+
expBytes, err := icatypes.ModuleCdc.MarshalJSON(&expMetadata)
215+
suite.Require().NoError(err)
216+
217+
suite.Require().Equal(version, string(expBytes))
205218
suite.Require().NoError(err)
206219
} else {
207220
suite.Require().Error(err)

modules/apps/27-interchain-accounts/host/ibc_module.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -34,8 +34,8 @@ func (im IBCModule) OnChanOpenInit(
3434
chanCap *capabilitytypes.Capability,
3535
counterparty channeltypes.Counterparty,
3636
version string,
37-
) error {
38-
return sdkerrors.Wrap(icatypes.ErrInvalidChannelFlow, "channel handshake must be initiated by controller chain")
37+
) (string, error) {
38+
return "", sdkerrors.Wrap(icatypes.ErrInvalidChannelFlow, "channel handshake must be initiated by controller chain")
3939
}
4040

4141
// OnChanOpenTry implements the IBCModule interface

modules/apps/29-fee/ibc_middleware.go

+32-12
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
package fee
22

33
import (
4+
"strings"
5+
46
sdk "github.com/cosmos/cosmos-sdk/types"
57
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
68
capabilitytypes "github.com/cosmos/cosmos-sdk/x/capability/types"
@@ -39,25 +41,44 @@ func (im IBCMiddleware) OnChanOpenInit(
3941
chanCap *capabilitytypes.Capability,
4042
counterparty channeltypes.Counterparty,
4143
version string,
42-
) error {
44+
) (string, error) {
4345
var versionMetadata types.Metadata
44-
if err := types.ModuleCdc.UnmarshalJSON([]byte(version), &versionMetadata); err != nil {
45-
// Since it is valid for fee version to not be specified, the above middleware version may be for a middleware
46-
// lower down in the stack. Thus, if it is not a fee version we pass the entire version string onto the underlying
47-
// application.
48-
return im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID,
49-
chanCap, counterparty, version)
46+
47+
if strings.TrimSpace(version) == "" {
48+
// default version
49+
versionMetadata = types.Metadata{
50+
FeeVersion: types.Version,
51+
AppVersion: "",
52+
}
53+
} else {
54+
if err := types.ModuleCdc.UnmarshalJSON([]byte(version), &versionMetadata); err != nil {
55+
// Since it is valid for fee version to not be specified, the above middleware version may be for a middleware
56+
// lower down in the stack. Thus, if it is not a fee version we pass the entire version string onto the underlying
57+
// application.
58+
return im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID,
59+
chanCap, counterparty, version)
60+
}
5061
}
5162

5263
if versionMetadata.FeeVersion != types.Version {
53-
return sdkerrors.Wrapf(types.ErrInvalidVersion, "expected %s, got %s", types.Version, versionMetadata.FeeVersion)
64+
return "", sdkerrors.Wrapf(types.ErrInvalidVersion, "expected %s, got %s", types.Version, versionMetadata.FeeVersion)
65+
}
66+
67+
appVersion, err := im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID, chanCap, counterparty, versionMetadata.AppVersion)
68+
if err != nil {
69+
return "", err
70+
}
71+
72+
versionMetadata.AppVersion = appVersion
73+
versionBytes, err := types.ModuleCdc.MarshalJSON(&versionMetadata)
74+
if err != nil {
75+
return "", err
5476
}
5577

5678
im.keeper.SetFeeEnabled(ctx, portID, channelID)
5779

5880
// call underlying app's OnChanOpenInit callback with the appVersion
59-
return im.app.OnChanOpenInit(ctx, order, connectionHops, portID, channelID,
60-
chanCap, counterparty, versionMetadata.AppVersion)
81+
return string(versionBytes), nil
6182
}
6283

6384
// OnChanOpenTry implements the IBCMiddleware interface
@@ -94,7 +115,6 @@ func (im IBCMiddleware) OnChanOpenTry(
94115
}
95116

96117
versionMetadata.AppVersion = appVersion
97-
98118
versionBytes, err := types.ModuleCdc.MarshalJSON(&versionMetadata)
99119
if err != nil {
100120
return "", err
@@ -116,7 +136,7 @@ func (im IBCMiddleware) OnChanOpenAck(
116136
if im.keeper.IsFeeEnabled(ctx, portID, channelID) {
117137
var versionMetadata types.Metadata
118138
if err := types.ModuleCdc.UnmarshalJSON([]byte(counterpartyVersion), &versionMetadata); err != nil {
119-
return sdkerrors.Wrap(types.ErrInvalidVersion, "failed to unmarshal ICS29 counterparty version metadata")
139+
return sdkerrors.Wrapf(err, "failed to unmarshal ICS29 counterparty version metadata: %s", counterpartyVersion)
120140
}
121141

122142
if versionMetadata.FeeVersion != types.Version {

modules/apps/29-fee/ibc_middleware_test.go

+35-7
Original file line numberDiff line numberDiff line change
@@ -25,34 +25,46 @@ var (
2525
// Tests OnChanOpenInit on ChainA
2626
func (suite *FeeTestSuite) TestOnChanOpenInit() {
2727
testCases := []struct {
28-
name string
29-
version string
30-
expPass bool
28+
name string
29+
version string
30+
expPass bool
31+
isFeeEnabled bool
3132
}{
3233
{
3334
"success - valid fee middleware and mock version",
3435
string(types.ModuleCdc.MustMarshalJSON(&types.Metadata{FeeVersion: types.Version, AppVersion: ibcmock.Version})),
3536
true,
37+
true,
3638
},
3739
{
3840
"success - fee version not included, only perform mock logic",
3941
ibcmock.Version,
4042
true,
43+
false,
4144
},
4245
{
4346
"invalid fee middleware version",
4447
string(types.ModuleCdc.MustMarshalJSON(&types.Metadata{FeeVersion: "invalid-ics29-1", AppVersion: ibcmock.Version})),
4548
false,
49+
false,
4650
},
4751
{
4852
"invalid mock version",
4953
string(types.ModuleCdc.MustMarshalJSON(&types.Metadata{FeeVersion: types.Version, AppVersion: "invalid-mock-version"})),
5054
false,
55+
false,
5156
},
5257
{
5358
"mock version not wrapped",
5459
types.Version,
5560
false,
61+
false,
62+
},
63+
{
64+
"passing an empty string returns default version",
65+
"",
66+
true,
67+
true,
5668
},
5769
}
5870

@@ -68,11 +80,11 @@ func (suite *FeeTestSuite) TestOnChanOpenInit() {
6880
suite.chainA.GetSimApp().FeeMockModule.IBCApp.OnChanOpenInit = func(ctx sdk.Context, order channeltypes.Order, connectionHops []string,
6981
portID, channelID string, chanCap *capabilitytypes.Capability,
7082
counterparty channeltypes.Counterparty, version string,
71-
) error {
83+
) (string, error) {
7284
if version != ibcmock.Version {
73-
return fmt.Errorf("incorrect mock version")
85+
return "", fmt.Errorf("incorrect mock version")
7486
}
75-
return nil
87+
return ibcmock.Version, nil
7688
}
7789

7890
suite.path.EndpointA.ChannelID = ibctesting.FirstChannelID
@@ -95,13 +107,29 @@ func (suite *FeeTestSuite) TestOnChanOpenInit() {
95107
cbs, ok := suite.chainA.App.GetIBCKeeper().Router.GetRoute(module)
96108
suite.Require().True(ok)
97109

98-
err = cbs.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
110+
version, err := cbs.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
99111
suite.path.EndpointA.ChannelConfig.PortID, suite.path.EndpointA.ChannelID, chanCap, counterparty, channel.Version)
100112

101113
if tc.expPass {
114+
// check if the channel is fee enabled. If so version string should include metaData
115+
if tc.isFeeEnabled {
116+
versionMetadata := types.Metadata{
117+
FeeVersion: types.Version,
118+
AppVersion: ibcmock.Version,
119+
}
120+
121+
versionBytes, err := types.ModuleCdc.MarshalJSON(&versionMetadata)
122+
suite.Require().NoError(err)
123+
124+
suite.Require().Equal(version, string(versionBytes))
125+
} else {
126+
suite.Require().Equal(ibcmock.Version, version)
127+
}
128+
102129
suite.Require().NoError(err, "unexpected error from version: %s", tc.version)
103130
} else {
104131
suite.Require().Error(err, "error not returned for version: %s", tc.version)
132+
suite.Require().Equal("", version)
105133
}
106134
})
107135
}

modules/apps/29-fee/transfer_test.go

-1
Original file line numberDiff line numberDiff line change
@@ -68,5 +68,4 @@ func (suite *FeeTestSuite) TestFeeTransfer() {
6868
fee.AckFee.Add(fee.TimeoutFee...), // ack fee paid, timeout fee refunded
6969
sdk.NewCoins(suite.chainA.GetSimApp().BankKeeper.GetBalance(suite.chainA.GetContext(), suite.chainA.SenderAccount.GetAddress(), ibctesting.TestCoin.Denom)).Sub(originalChainASenderAccountBalance),
7070
)
71-
7271
}

modules/apps/transfer/ibc_module.go

+10-5
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ package transfer
33
import (
44
"fmt"
55
"math"
6+
"strings"
67

78
sdk "github.com/cosmos/cosmos-sdk/types"
89
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
@@ -70,21 +71,25 @@ func (im IBCModule) OnChanOpenInit(
7071
chanCap *capabilitytypes.Capability,
7172
counterparty channeltypes.Counterparty,
7273
version string,
73-
) error {
74+
) (string, error) {
7475
if err := ValidateTransferChannelParams(ctx, im.keeper, order, portID, channelID); err != nil {
75-
return err
76+
return "", err
77+
}
78+
79+
if strings.TrimSpace(version) == "" {
80+
version = types.Version
7681
}
7782

7883
if version != types.Version {
79-
return sdkerrors.Wrapf(types.ErrInvalidVersion, "got %s, expected %s", version, types.Version)
84+
return "", sdkerrors.Wrapf(types.ErrInvalidVersion, "got %s, expected %s", version, types.Version)
8085
}
8186

8287
// Claim channel capability passed back by IBC module
8388
if err := im.keeper.ClaimCapability(ctx, chanCap, host.ChannelCapabilityPath(portID, channelID)); err != nil {
84-
return err
89+
return "", err
8590
}
8691

87-
return nil
92+
return version, nil
8893
}
8994

9095
// OnChanOpenTry implements the IBCModule interface.

modules/apps/transfer/ibc_module_test.go

+11-7
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import (
55

66
capabilitytypes "github.com/cosmos/cosmos-sdk/x/capability/types"
77

8+
"github.com/cosmos/ibc-go/v3/modules/apps/transfer"
89
"github.com/cosmos/ibc-go/v3/modules/apps/transfer/types"
910
channeltypes "github.com/cosmos/ibc-go/v3/modules/core/04-channel/types"
1011
host "github.com/cosmos/ibc-go/v3/modules/core/24-host"
@@ -28,6 +29,11 @@ func (suite *TransferTestSuite) TestOnChanOpenInit() {
2829
{
2930
"success", func() {}, true,
3031
},
32+
{
33+
"empty version string", func() {
34+
channel.Version = ""
35+
}, true,
36+
},
3137
{
3238
"max channels reached", func() {
3339
path.EndpointA.ChannelID = channeltypes.FormatChannelIdentifier(math.MaxUint32 + 1)
@@ -74,25 +80,23 @@ func (suite *TransferTestSuite) TestOnChanOpenInit() {
7480
Version: types.Version,
7581
}
7682

77-
module, _, err := suite.chainA.App.GetIBCKeeper().PortKeeper.LookupModuleByPort(suite.chainA.GetContext(), ibctesting.TransferPort)
78-
suite.Require().NoError(err)
79-
83+
var err error
8084
chanCap, err = suite.chainA.App.GetScopedIBCKeeper().NewCapability(suite.chainA.GetContext(), host.ChannelCapabilityPath(ibctesting.TransferPort, path.EndpointA.ChannelID))
8185
suite.Require().NoError(err)
8286

83-
cbs, ok := suite.chainA.App.GetIBCKeeper().Router.GetRoute(module)
84-
suite.Require().True(ok)
85-
8687
tc.malleate() // explicitly change fields in channel and testChannel
8788

88-
err = cbs.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
89+
transferModule := transfer.NewIBCModule(suite.chainA.GetSimApp().TransferKeeper)
90+
version, err := transferModule.OnChanOpenInit(suite.chainA.GetContext(), channel.Ordering, channel.GetConnectionHops(),
8991
path.EndpointA.ChannelConfig.PortID, path.EndpointA.ChannelID, chanCap, counterparty, channel.GetVersion(),
9092
)
9193

9294
if tc.expPass {
9395
suite.Require().NoError(err)
96+
suite.Require().Equal(types.Version, version)
9497
} else {
9598
suite.Require().Error(err)
99+
suite.Require().Equal(version, "")
96100
}
97101

98102
})

0 commit comments

Comments
 (0)