Skip to content

Commit bcd74b7

Browse files
mergify[bot]damiannolancolin-axner
authored
imp: add updateClientCheckTx to redunant relayer ante decorator (backport #6279) (#6305)
* imp: add updateClientCheckTx to redunant relayer ante decorator (#6279) * imp: add checkTxUpdateClient to redunant relayer ante decorator * chore: update godoc and duplicate imports * test: add coverage for checkTxUpdateClient func * chore: rename ante func to updateClientCheckTx (cherry picked from commit 3da4830) * Update modules/core/ante/ante.go Co-authored-by: colin axnér <[email protected]> --------- Co-authored-by: Damian Nolan <[email protected]> Co-authored-by: colin axnér <[email protected]>
1 parent e18de37 commit bcd74b7

File tree

3 files changed

+139
-2
lines changed

3 files changed

+139
-2
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ Ref: https://keepachangelog.com/en/1.0.0/
4545
### Improvements
4646
* (core/ante) [\#6302](https://github.com/cosmos/ibc-go/pull/6302) Performance: Skip app callbacks during RecvPacket execution in checkTx within the redundant relay ante handler.
4747
* (core/ante) [\#6280](https://github.com/cosmos/ibc-go/pull/6280) Performance: Skip redundant proof checking in RecvPacket execution in reCheckTx within the redundant relay ante handler.
48+
* (core/ante) [\#6306](https://github.com/cosmos/ibc-go/pull/6306) Performance: Skip misbehaviour checks in UpdateClient flow and skip signature checks in reCheckTx mode.
4849

4950
### Features
5051

modules/core/ante/ante.go

+44-2
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,10 @@ import (
77

88
clienttypes "github.com/cosmos/ibc-go/v8/modules/core/02-client/types"
99
channeltypes "github.com/cosmos/ibc-go/v8/modules/core/04-channel/types"
10+
"github.com/cosmos/ibc-go/v8/modules/core/exported"
1011
"github.com/cosmos/ibc-go/v8/modules/core/keeper"
12+
solomachine "github.com/cosmos/ibc-go/v8/modules/light-clients/06-solomachine"
13+
tendermint "github.com/cosmos/ibc-go/v8/modules/light-clients/07-tendermint"
1114
)
1215

1316
type RedundantRelayDecorator struct {
@@ -84,8 +87,7 @@ func (rrd RedundantRelayDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simula
8487
packetMsgs++
8588

8689
case *clienttypes.MsgUpdateClient:
87-
_, err := rrd.k.UpdateClient(ctx, msg)
88-
if err != nil {
90+
if err := rrd.updateClientCheckTx(ctx, msg); err != nil {
8991
return ctx, err
9092
}
9193

@@ -105,6 +107,46 @@ func (rrd RedundantRelayDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simula
105107
return next(ctx, tx, simulate)
106108
}
107109

110+
// updateClientCheckTx runs a subset of ibc client update logic to be used specifically within the RedundantRelayDecorator AnteHandler.
111+
// The following function performs ibc client message verification for CheckTx only and state updates in both CheckTx and ReCheckTx.
112+
// Note that misbehaviour checks are omitted.
113+
func (rrd RedundantRelayDecorator) updateClientCheckTx(ctx sdk.Context, msg *clienttypes.MsgUpdateClient) error {
114+
clientMsg, err := clienttypes.UnpackClientMessage(msg.ClientMessage)
115+
if err != nil {
116+
return err
117+
}
118+
119+
clientState, found := rrd.k.ClientKeeper.GetClientState(ctx, msg.ClientId)
120+
if !found {
121+
return errorsmod.Wrapf(clienttypes.ErrClientNotFound, msg.ClientId)
122+
}
123+
124+
if status := rrd.k.ClientKeeper.GetClientStatus(ctx, clientState, msg.ClientId); status != exported.Active {
125+
return errorsmod.Wrapf(clienttypes.ErrClientNotActive, "cannot update client (%s) with status %s", msg.ClientId, status)
126+
}
127+
128+
clientStore := rrd.k.ClientKeeper.ClientStore(ctx, msg.ClientId)
129+
130+
if !ctx.IsReCheckTx() {
131+
if err := clientState.VerifyClientMessage(ctx, rrd.k.Codec(), clientStore, clientMsg); err != nil {
132+
return err
133+
}
134+
}
135+
136+
// NOTE: the following avoids panics in ante handler client updates for ibc-go v8.3.x
137+
// without state machine breaking changes within light client modules.
138+
switch clientMsg.(type) {
139+
case *solomachine.Misbehaviour:
140+
// ignore solomachine misbehaviour for update state in ante
141+
case *tendermint.Misbehaviour:
142+
// ignore tendermint misbehaviour for update state in ante
143+
default:
144+
heights := clientState.UpdateState(ctx, rrd.k.Codec(), clientStore, clientMsg)
145+
ctx.Logger().With("module", "x/"+exported.ModuleName).Debug("ante ibc client update", "consensusHeights", heights)
146+
}
147+
return nil
148+
}
149+
108150
// recvPacketCheckTx runs a subset of ibc recv packet logic to be used specifically within the RedundantRelayDecorator AnteHandler.
109151
// It only performs core IBC receiving logic and skips any application logic.
110152
func (rrd RedundantRelayDecorator) recvPacketCheckTx(ctx sdk.Context, msg *channeltypes.MsgRecvPacket) (*channeltypes.MsgRecvPacketResponse, error) {

modules/core/ante/ante_test.go

+94
Original file line numberDiff line numberDiff line change
@@ -3,17 +3,20 @@ package ante_test
33
import (
44
"fmt"
55
"testing"
6+
"time"
67

78
"github.com/stretchr/testify/require"
89
testifysuite "github.com/stretchr/testify/suite"
910

11+
codectypes "github.com/cosmos/cosmos-sdk/codec/types"
1012
sdk "github.com/cosmos/cosmos-sdk/types"
1113

1214
clienttypes "github.com/cosmos/ibc-go/v8/modules/core/02-client/types"
1315
channeltypes "github.com/cosmos/ibc-go/v8/modules/core/04-channel/types"
1416
host "github.com/cosmos/ibc-go/v8/modules/core/24-host"
1517
"github.com/cosmos/ibc-go/v8/modules/core/ante"
1618
"github.com/cosmos/ibc-go/v8/modules/core/exported"
19+
ibctm "github.com/cosmos/ibc-go/v8/modules/light-clients/07-tendermint"
1720
ibctesting "github.com/cosmos/ibc-go/v8/testing"
1821
)
1922

@@ -357,6 +360,64 @@ func (suite *AnteTestSuite) TestAnteDecoratorCheckTx() {
357360
},
358361
true,
359362
},
363+
{
364+
"success on new UpdateClient messages: solomachine misbehaviour",
365+
func(suite *AnteTestSuite) []sdk.Msg {
366+
solomachine := ibctesting.NewSolomachine(suite.T(), suite.chainB.Codec, "06-solomachine-0", "testing", 1)
367+
suite.chainB.GetSimApp().GetIBCKeeper().ClientKeeper.SetClientState(suite.chainB.GetContext(), solomachine.ClientID, solomachine.ClientState())
368+
369+
msgUpdateClient, err := clienttypes.NewMsgUpdateClient(solomachine.ClientID, solomachine.CreateMisbehaviour(), suite.chainB.SenderAccount.GetAddress().String())
370+
suite.Require().NoError(err)
371+
372+
msgs := []sdk.Msg{msgUpdateClient}
373+
374+
return msgs
375+
},
376+
true,
377+
},
378+
{
379+
"success on new UpdateClient messages: solomachine multisig misbehaviour",
380+
func(suite *AnteTestSuite) []sdk.Msg {
381+
solomachine := ibctesting.NewSolomachine(suite.T(), suite.chainA.Codec, "06-solomachine-0", "testing", 4)
382+
suite.chainB.GetSimApp().GetIBCKeeper().ClientKeeper.SetClientState(suite.chainB.GetContext(), solomachine.ClientID, solomachine.ClientState())
383+
384+
msgUpdateClient, err := clienttypes.NewMsgUpdateClient(solomachine.ClientID, solomachine.CreateMisbehaviour(), suite.chainB.SenderAccount.GetAddress().String())
385+
suite.Require().NoError(err)
386+
387+
msgs := []sdk.Msg{msgUpdateClient}
388+
389+
return msgs
390+
},
391+
true,
392+
},
393+
{
394+
"success on new UpdateClient messages: tendermint misbehaviour",
395+
func(suite *AnteTestSuite) []sdk.Msg {
396+
trustedHeight := suite.path.EndpointB.GetClientState().GetLatestHeight().(clienttypes.Height)
397+
398+
trustedVals, found := suite.chainA.GetValsAtHeight(int64(trustedHeight.RevisionHeight) + 1)
399+
suite.Require().True(found)
400+
401+
err := suite.path.EndpointB.UpdateClient()
402+
suite.Require().NoError(err)
403+
404+
height := suite.path.EndpointB.GetClientState().GetLatestHeight().(clienttypes.Height)
405+
406+
// construct valid fork misbehaviour: two headers at the same height with different time
407+
misbehaviour := &ibctm.Misbehaviour{
408+
Header1: suite.chainA.CreateTMClientHeader(suite.chainA.ChainID, int64(height.RevisionHeight), trustedHeight, suite.chainA.CurrentHeader.Time.Add(time.Minute), suite.chainA.Vals, suite.chainA.NextVals, trustedVals, suite.chainA.Signers),
409+
Header2: suite.chainA.CreateTMClientHeader(suite.chainA.ChainID, int64(height.RevisionHeight), trustedHeight, suite.chainA.CurrentHeader.Time, suite.chainA.Vals, suite.chainA.NextVals, trustedVals, suite.chainA.Signers),
410+
}
411+
412+
msgUpdateClient, err := clienttypes.NewMsgUpdateClient(suite.path.EndpointB.ClientID, misbehaviour, suite.chainB.SenderAccount.GetAddress().String())
413+
suite.Require().NoError(err)
414+
415+
msgs := []sdk.Msg{msgUpdateClient}
416+
417+
return msgs
418+
},
419+
true,
420+
},
360421
{
361422
"no success on one redundant RecvPacket message",
362423
func(suite *AnteTestSuite) []sdk.Msg {
@@ -402,6 +463,39 @@ func (suite *AnteTestSuite) TestAnteDecoratorCheckTx() {
402463
},
403464
false,
404465
},
466+
{
467+
"no success on one new UpdateClient message: invalid client identifier",
468+
func(suite *AnteTestSuite) []sdk.Msg {
469+
clientMsg, err := codectypes.NewAnyWithValue(&ibctm.Header{})
470+
suite.Require().NoError(err)
471+
472+
msgs := []sdk.Msg{&clienttypes.MsgUpdateClient{ClientId: ibctesting.InvalidID, ClientMessage: clientMsg}}
473+
return msgs
474+
},
475+
false,
476+
},
477+
{
478+
"no success on one new UpdateClient message: client module not found",
479+
func(suite *AnteTestSuite) []sdk.Msg {
480+
clientMsg, err := codectypes.NewAnyWithValue(&ibctm.Header{})
481+
suite.Require().NoError(err)
482+
483+
msgs := []sdk.Msg{&clienttypes.MsgUpdateClient{ClientId: clienttypes.FormatClientIdentifier("08-wasm", 1), ClientMessage: clientMsg}}
484+
return msgs
485+
},
486+
false,
487+
},
488+
{
489+
"no success on one new UpdateClient message: no consensus state for trusted height",
490+
func(suite *AnteTestSuite) []sdk.Msg {
491+
clientMsg, err := codectypes.NewAnyWithValue(&ibctm.Header{TrustedHeight: clienttypes.NewHeight(1, 10000)})
492+
suite.Require().NoError(err)
493+
494+
msgs := []sdk.Msg{&clienttypes.MsgUpdateClient{ClientId: suite.path.EndpointA.ClientID, ClientMessage: clientMsg}}
495+
return msgs
496+
},
497+
false,
498+
},
405499
{
406500
"no success on three new UpdateClient messages and three redundant messages of each type",
407501
func(suite *AnteTestSuite) []sdk.Msg {

0 commit comments

Comments
 (0)