From 5b39f18b543692d35ab9b45b64bcd538fc862065 Mon Sep 17 00:00:00 2001 From: Damian Nolan Date: Wed, 21 Dec 2022 11:13:47 +0000 Subject: [PATCH 1/3] move pruning to above duplicate update check --- modules/light-clients/07-tendermint/update.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/light-clients/07-tendermint/update.go b/modules/light-clients/07-tendermint/update.go index ecd1f58e21e..e8045e144d8 100644 --- a/modules/light-clients/07-tendermint/update.go +++ b/modules/light-clients/07-tendermint/update.go @@ -134,14 +134,14 @@ func (cs ClientState) UpdateState(ctx sdk.Context, cdc codec.BinaryCodec, client panic(fmt.Errorf("expected type %T, got %T", &Header{}, clientMsg)) } + cs.pruneOldestConsensusState(ctx, cdc, clientStore) + // check for duplicate update if consensusState, _ := GetConsensusState(clientStore, cdc, header.GetHeight()); consensusState != nil { // perform no-op return []exported.Height{header.GetHeight()} } - cs.pruneOldestConsensusState(ctx, cdc, clientStore) - height := header.GetHeight().(clienttypes.Height) if height.GT(cs.LatestHeight) { cs.LatestHeight = height From 7f02b5b84be0159ed75c8162c6b6776892072fd7 Mon Sep 17 00:00:00 2001 From: Damian Nolan Date: Wed, 21 Dec 2022 12:23:13 +0000 Subject: [PATCH 2/3] adding test for pruning on duplicate header update --- .../07-tendermint/update_test.go | 37 +++++++++++++++++++ 1 file changed, 37 insertions(+) diff --git a/modules/light-clients/07-tendermint/update_test.go b/modules/light-clients/07-tendermint/update_test.go index 7da6ba3cf2c..4c71c5f17cc 100644 --- a/modules/light-clients/07-tendermint/update_test.go +++ b/modules/light-clients/07-tendermint/update_test.go @@ -420,6 +420,43 @@ func (suite *TendermintTestSuite) TestUpdateState() { suite.Require().False(found) }, true, }, + { + "success with pruned consensus state using duplicate header", func() { + // this height will be expired and pruned + err := path.EndpointA.UpdateClient() + suite.Require().NoError(err) + pruneHeight = path.EndpointA.GetClientState().GetLatestHeight().(clienttypes.Height) + + // Increment the time by a week + suite.coordinator.IncrementTimeBy(7 * 24 * time.Hour) + + // create the consensus state that can be used as trusted height for next update + err = path.EndpointA.UpdateClient() + suite.Require().NoError(err) + + // Increment the time by another week, then update the client. + // This will cause the first two consensus states to become expired. + suite.coordinator.IncrementTimeBy(7 * 24 * time.Hour) + err = path.EndpointA.UpdateClient() + suite.Require().NoError(err) + + // use the same header which just updated the client + clientMessage, err = path.EndpointA.Chain.ConstructUpdateTMClientHeader(path.EndpointA.Counterparty.Chain, path.EndpointA.ClientID) + suite.Require().NoError(err) + }, + func() { + tmHeader, ok := clientMessage.(*ibctm.Header) + suite.Require().True(ok) + + clientState := path.EndpointA.GetClientState() + suite.Require().True(clientState.GetLatestHeight().EQ(tmHeader.GetHeight())) // new update, updated client state should have changed + suite.Require().True(clientState.GetLatestHeight().EQ(consensusHeights[0])) + + // ensure consensus state was pruned + _, found := path.EndpointA.Chain.GetConsensusState(path.EndpointA.ClientID, pruneHeight) + suite.Require().False(found) + }, true, + }, { "invalid ClientMessage type", func() { clientMessage = &ibctm.Misbehaviour{} From 3ae3d4778a3cd0582c0bc3a5a2c1bd57c2a75304 Mon Sep 17 00:00:00 2001 From: Damian Nolan Date: Wed, 21 Dec 2022 12:43:50 +0000 Subject: [PATCH 3/3] adding additional check - assert that a consensus state exists at the prune height --- modules/light-clients/07-tendermint/update_test.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/modules/light-clients/07-tendermint/update_test.go b/modules/light-clients/07-tendermint/update_test.go index 4c71c5f17cc..09d5fc6afe9 100644 --- a/modules/light-clients/07-tendermint/update_test.go +++ b/modules/light-clients/07-tendermint/update_test.go @@ -427,6 +427,11 @@ func (suite *TendermintTestSuite) TestUpdateState() { suite.Require().NoError(err) pruneHeight = path.EndpointA.GetClientState().GetLatestHeight().(clienttypes.Height) + // assert that a consensus state exists at the prune height + consensusState, found := path.EndpointA.Chain.GetConsensusState(path.EndpointA.ClientID, pruneHeight) + suite.Require().True(found) + suite.Require().NotNil(consensusState) + // Increment the time by a week suite.coordinator.IncrementTimeBy(7 * 24 * time.Hour)