@@ -508,3 +508,159 @@ func (suite *KeeperTestSuite) TestUpdateClientEventEmission() {
508
508
}
509
509
suite .Require ().True (contains )
510
510
}
511
+
512
+ func (suite * KeeperTestSuite ) TestRecoverClient () {
513
+ var (
514
+ subject , substitute string
515
+ subjectClientState , substituteClientState exported.ClientState
516
+ )
517
+
518
+ testCases := []struct {
519
+ msg string
520
+ malleate func ()
521
+ expErr error
522
+ }{
523
+ {
524
+ "success" ,
525
+ func () {},
526
+ nil ,
527
+ },
528
+ {
529
+ "success, subject and substitute use different revision number" ,
530
+ func () {
531
+ tmClientState , ok := substituteClientState .(* ibctm.ClientState )
532
+ suite .Require ().True (ok )
533
+ consState , found := suite .chainA .App .GetIBCKeeper ().ClientKeeper .GetClientConsensusState (suite .chainA .GetContext (), substitute , tmClientState .LatestHeight )
534
+ suite .Require ().True (found )
535
+ newRevisionNumber := tmClientState .GetLatestHeight ().GetRevisionNumber () + 1
536
+
537
+ tmClientState .LatestHeight = clienttypes .NewHeight (newRevisionNumber , tmClientState .GetLatestHeight ().GetRevisionHeight ())
538
+
539
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientConsensusState (suite .chainA .GetContext (), substitute , tmClientState .LatestHeight , consState )
540
+ clientStore := suite .chainA .App .GetIBCKeeper ().ClientKeeper .ClientStore (suite .chainA .GetContext (), substitute )
541
+ ibctm .SetProcessedTime (clientStore , tmClientState .LatestHeight , 100 )
542
+ ibctm .SetProcessedHeight (clientStore , tmClientState .LatestHeight , clienttypes .NewHeight (0 , 1 ))
543
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), substitute , tmClientState )
544
+ },
545
+ nil ,
546
+ },
547
+ {
548
+ "subject client does not exist" ,
549
+ func () {
550
+ subject = ibctesting .InvalidID
551
+ },
552
+ clienttypes .ErrClientNotFound ,
553
+ },
554
+ {
555
+ "subject is Active" ,
556
+ func () {
557
+ tmClientState , ok := subjectClientState .(* ibctm.ClientState )
558
+ suite .Require ().True (ok )
559
+ // Set FrozenHeight to zero to ensure client is reported as Active
560
+ tmClientState .FrozenHeight = clienttypes .ZeroHeight ()
561
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), subject , tmClientState )
562
+ },
563
+ clienttypes .ErrInvalidRecoveryClient ,
564
+ },
565
+ {
566
+ "substitute client does not exist" ,
567
+ func () {
568
+ substitute = ibctesting .InvalidID
569
+ },
570
+ clienttypes .ErrClientNotFound ,
571
+ },
572
+ {
573
+ "subject and substitute have equal latest height" ,
574
+ func () {
575
+ tmClientState , ok := subjectClientState .(* ibctm.ClientState )
576
+ suite .Require ().True (ok )
577
+ tmClientState .LatestHeight = substituteClientState .GetLatestHeight ().(clienttypes.Height )
578
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), subject , tmClientState )
579
+ },
580
+ clienttypes .ErrInvalidHeight ,
581
+ },
582
+ {
583
+ "subject height is greater than substitute height" ,
584
+ func () {
585
+ tmClientState , ok := subjectClientState .(* ibctm.ClientState )
586
+ suite .Require ().True (ok )
587
+ tmClientState .LatestHeight = substituteClientState .GetLatestHeight ().Increment ().(clienttypes.Height )
588
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), subject , tmClientState )
589
+ },
590
+ clienttypes .ErrInvalidHeight ,
591
+ },
592
+ {
593
+ "substitute is frozen" ,
594
+ func () {
595
+ tmClientState , ok := substituteClientState .(* ibctm.ClientState )
596
+ suite .Require ().True (ok )
597
+ tmClientState .FrozenHeight = clienttypes .NewHeight (0 , 1 )
598
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), substitute , tmClientState )
599
+ },
600
+ clienttypes .ErrClientNotActive ,
601
+ },
602
+ {
603
+ "CheckSubstituteAndUpdateState fails, substitute client trust level doesn't match subject client trust level" ,
604
+ func () {
605
+ tmClientState , ok := substituteClientState .(* ibctm.ClientState )
606
+ suite .Require ().True (ok )
607
+ tmClientState .UnbondingPeriod += time .Minute
608
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), substitute , tmClientState )
609
+ },
610
+ clienttypes .ErrInvalidSubstitute ,
611
+ },
612
+ }
613
+
614
+ for _ , tc := range testCases {
615
+ tc := tc
616
+
617
+ suite .Run (tc .msg , func () {
618
+ suite .SetupTest () // reset
619
+
620
+ subjectPath := ibctesting .NewPath (suite .chainA , suite .chainB )
621
+ suite .coordinator .SetupClients (subjectPath )
622
+ subject = subjectPath .EndpointA .ClientID
623
+ subjectClientState = suite .chainA .GetClientState (subject )
624
+
625
+ substitutePath := ibctesting .NewPath (suite .chainA , suite .chainB )
626
+ suite .coordinator .SetupClients (substitutePath )
627
+ substitute = substitutePath .EndpointA .ClientID
628
+
629
+ // update substitute twice
630
+ err := substitutePath .EndpointA .UpdateClient ()
631
+ suite .Require ().NoError (err )
632
+ err = substitutePath .EndpointA .UpdateClient ()
633
+ suite .Require ().NoError (err )
634
+ substituteClientState = suite .chainA .GetClientState (substitute )
635
+
636
+ tmClientState , ok := subjectClientState .(* ibctm.ClientState )
637
+ suite .Require ().True (ok )
638
+ tmClientState .AllowUpdateAfterMisbehaviour = true
639
+ tmClientState .AllowUpdateAfterExpiry = true
640
+ tmClientState .FrozenHeight = tmClientState .LatestHeight
641
+ suite .chainA .App .GetIBCKeeper ().ClientKeeper .SetClientState (suite .chainA .GetContext (), subject , tmClientState )
642
+
643
+ tmClientState , ok = substituteClientState .(* ibctm.ClientState )
644
+ suite .Require ().True (ok )
645
+ tmClientState .AllowUpdateAfterMisbehaviour = true
646
+ tmClientState .AllowUpdateAfterExpiry = true
647
+
648
+ tc .malleate ()
649
+
650
+ err = suite .chainA .App .GetIBCKeeper ().ClientKeeper .RecoverClient (suite .chainA .GetContext (), subject , substitute )
651
+
652
+ expPass := tc .expErr == nil
653
+ if expPass {
654
+ suite .Require ().NoError (err )
655
+
656
+ // Assert that client status is now Active
657
+ clientStore := suite .chainA .App .GetIBCKeeper ().ClientKeeper .ClientStore (suite .chainA .GetContext (), subjectPath .EndpointA .ClientID )
658
+ tmClientState := subjectPath .EndpointA .GetClientState ().(* ibctm.ClientState )
659
+ suite .Require ().Equal (tmClientState .Status (suite .chainA .GetContext (), clientStore , suite .chainA .App .AppCodec ()), exported .Active )
660
+ } else {
661
+ suite .Require ().Error (err )
662
+ suite .Require ().ErrorIs (err , tc .expErr )
663
+ }
664
+ })
665
+ }
666
+ }
0 commit comments