@@ -887,9 +887,9 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFAfterEndOfList(t *testing.T) {
887
887
}()
888
888
889
889
servers := newHangingServerGroup (t , 3 )
890
- defer servers .close ()
891
890
rb := manual .NewBuilderWithScheme ("whatever" )
892
- rb .InitialState (resolver.State {Addresses : servers .addrs })
891
+ addrs := resolverAddrsFromHangingServers (servers )
892
+ rb .InitialState (resolver.State {Addresses : addrs })
893
893
cc , err := grpc .NewClient ("whatever:///this-gets-overwritten" ,
894
894
grpc .WithTransportCredentials (insecure .NewCredentials ()),
895
895
grpc .WithDefaultServiceConfig (fmt .Sprintf (`{"loadBalancingConfig": [{"%s":{}}]}` , pickfirstleaf .Name )),
@@ -904,37 +904,37 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFAfterEndOfList(t *testing.T) {
904
904
testutils .AwaitState (ctx , t , cc , connectivity .Connecting )
905
905
906
906
// Verify that only the first server is contacted.
907
- if err := servers .awaitContacted (ctx , 0 ); err != nil {
908
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [0 ], err )
907
+ if err := servers [ 0 ] .awaitContacted (ctx ); err != nil {
908
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [0 ], err )
909
909
}
910
910
911
911
// Ensure no other servers are contacted.
912
- if got , want := servers .isContacted (1 ), false ; got != want {
913
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [1 ], got , want )
912
+ if got , want := servers [ 1 ] .isContacted (), false ; got != want {
913
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [1 ], got , want )
914
914
}
915
- if got , want := servers .isContacted (2 ), false ; got != want {
916
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [2 ], got , want )
915
+ if got , want := servers [ 2 ] .isContacted (), false ; got != want {
916
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [2 ], got , want )
917
917
}
918
918
919
919
// Make the happy eyeballs timer fire twice so that pickfirst reaches the
920
920
// last address in the list.
921
921
timerCh <- struct {}{}
922
922
923
923
// Verify that the second server is contacted and 3rd isn't.
924
- if err := servers .awaitContacted (ctx , 1 ); err != nil {
925
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [1 ], err )
924
+ if err := servers [ 1 ] .awaitContacted (ctx ); err != nil {
925
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [1 ], err )
926
926
}
927
927
928
- if got , want := servers .isContacted (2 ), false ; got != want {
929
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [2 ], got , want )
928
+ if got , want := servers [ 2 ] .isContacted (), false ; got != want {
929
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [2 ], got , want )
930
930
}
931
931
timerCh <- struct {}{}
932
- if err := servers .awaitContacted (ctx , 2 ); err != nil {
933
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [2 ], err )
932
+ if err := servers [ 2 ] .awaitContacted (ctx ); err != nil {
933
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [2 ], err )
934
934
}
935
935
936
936
// First SubConn Fails.
937
- servers .closeConn (0 )
937
+ servers [ 0 ] .closeConn ()
938
938
939
939
// No TF should be reported until the first pass is complete.
940
940
shortCtx , shortCancel := context .WithTimeout (ctx , defaultTestShortTimeout )
@@ -947,12 +947,12 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFAfterEndOfList(t *testing.T) {
947
947
timerCh <- struct {}{}
948
948
949
949
// Third SubConn fails.
950
- servers .closeConn (2 )
950
+ servers [ 2 ] .closeConn ()
951
951
952
952
testutils .AwaitNotState (shortCtx , t , cc , connectivity .TransientFailure )
953
953
954
954
// Last SubConn fails, this should result in a TF update.
955
- servers .closeConn (1 )
955
+ servers [ 1 ] .closeConn ()
956
956
testutils .AwaitState (ctx , t , cc , connectivity .TransientFailure )
957
957
}
958
958
@@ -988,9 +988,9 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TriggerConnectionDelay(t *testing.T) {
988
988
}()
989
989
990
990
servers := newHangingServerGroup (t , 2 )
991
- defer servers . close ( )
991
+ addrs := resolverAddrsFromHangingServers ( servers )
992
992
rb := manual .NewBuilderWithScheme ("whatever" )
993
- rb .InitialState (resolver.State {Addresses : servers . addrs })
993
+ rb .InitialState (resolver.State {Addresses : addrs })
994
994
cc , err := grpc .NewClient ("whatever:///this-gets-overwritten" ,
995
995
grpc .WithTransportCredentials (insecure .NewCredentials ()),
996
996
grpc .WithDefaultServiceConfig (fmt .Sprintf (`{"loadBalancingConfig": [{"%s":{}}]}` , pickfirstleaf .Name )),
@@ -1005,21 +1005,21 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TriggerConnectionDelay(t *testing.T) {
1005
1005
testutils .AwaitState (ctx , t , cc , connectivity .Connecting )
1006
1006
1007
1007
// Verify that the first server is contacted.
1008
- if err := servers .awaitContacted (ctx , 0 ); err != nil {
1009
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [0 ], err )
1008
+ if err := servers [ 0 ] .awaitContacted (ctx ); err != nil {
1009
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [0 ], err )
1010
1010
}
1011
1011
1012
- if got , want := servers .isContacted (1 ), false ; got != want {
1013
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [1 ], got , want )
1012
+ if got , want := servers [ 1 ] .isContacted (), false ; got != want {
1013
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [1 ], got , want )
1014
1014
}
1015
1015
1016
1016
timerCh <- struct {}{}
1017
1017
1018
1018
// Second connection attempt is successful.
1019
- if err := servers .awaitContacted (ctx , 1 ); err != nil {
1020
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [1 ], err )
1019
+ if err := servers [ 1 ] .awaitContacted (ctx ); err != nil {
1020
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [1 ], err )
1021
1021
}
1022
- servers .enterReady (1 )
1022
+ servers [ 1 ] .enterReady ()
1023
1023
testutils .AwaitState (ctx , t , cc , connectivity .Ready )
1024
1024
}
1025
1025
@@ -1055,9 +1055,9 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFThenTimerFires(t *testing.T) {
1055
1055
}()
1056
1056
1057
1057
servers := newHangingServerGroup (t , 3 )
1058
- defer servers . close ( )
1058
+ addrs := resolverAddrsFromHangingServers ( servers )
1059
1059
rb := manual .NewBuilderWithScheme ("whatever" )
1060
- rb .InitialState (resolver.State {Addresses : servers . addrs })
1060
+ rb .InitialState (resolver.State {Addresses : addrs })
1061
1061
cc , err := grpc .NewClient ("whatever:///this-gets-overwritten" ,
1062
1062
grpc .WithTransportCredentials (insecure .NewCredentials ()),
1063
1063
grpc .WithDefaultServiceConfig (fmt .Sprintf (`{"loadBalancingConfig": [{"%s":{}}]}` , pickfirstleaf .Name )),
@@ -1072,16 +1072,16 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFThenTimerFires(t *testing.T) {
1072
1072
testutils .AwaitState (ctx , t , cc , connectivity .Connecting )
1073
1073
1074
1074
// Verify that only the first server is contacted.
1075
- if err := servers .awaitContacted (ctx , 0 ); err != nil {
1076
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [0 ], err )
1075
+ if err := servers [ 0 ] .awaitContacted (ctx ); err != nil {
1076
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [0 ], err )
1077
1077
}
1078
1078
1079
1079
// Ensure no other servers are contacted.
1080
- if got , want := servers .isContacted (1 ), false ; got != want {
1081
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [1 ], got , want )
1080
+ if got , want := servers [ 1 ] .isContacted (), false ; got != want {
1081
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [1 ], got , want )
1082
1082
}
1083
- if got , want := servers .isContacted (2 ), false ; got != want {
1084
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [2 ], got , want )
1083
+ if got , want := servers [ 2 ] .isContacted (), false ; got != want {
1084
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [2 ], got , want )
1085
1085
}
1086
1086
1087
1087
// First SubConn Fails.
@@ -1090,29 +1090,29 @@ func (s) TestPickFirstLeaf_HappyEyeballs_TFThenTimerFires(t *testing.T) {
1090
1090
timerMu .Lock ()
1091
1091
timerCh = make (chan struct {})
1092
1092
timerMu .Unlock ()
1093
- servers .closeConn (0 )
1093
+ servers [ 0 ] .closeConn ()
1094
1094
1095
1095
// The second server is contacted.
1096
1096
// Verify that only the first server is contacted.
1097
- if err := servers .awaitContacted (ctx , 1 ); err != nil {
1098
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [1 ], err )
1097
+ if err := servers [ 1 ] .awaitContacted (ctx ); err != nil {
1098
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [1 ], err )
1099
1099
}
1100
1100
1101
1101
// Ensure no other servers are contacted.
1102
- if got , want := servers .isContacted (2 ), false ; got != want {
1103
- t .Fatalf ("Servers .isContacted(%q ) = %t, want %t" , servers . addrs [2 ], got , want )
1102
+ if got , want := servers [ 2 ] .isContacted (), false ; got != want {
1103
+ t .Fatalf ("Server[%q] .isContacted() = %t, want %t" , addrs [2 ], got , want )
1104
1104
}
1105
1105
1106
1106
// The happy eyeballs timer expires, skipping server[1] and requesting the creation
1107
1107
// of a third SubConn.
1108
1108
timerCh <- struct {}{}
1109
1109
1110
- if err := servers .awaitContacted (ctx , 2 ); err != nil {
1111
- t .Fatalf ("Server with address %q not contacted: %v" , servers . addrs [2 ], err )
1110
+ if err := servers [ 2 ] .awaitContacted (ctx ); err != nil {
1111
+ t .Fatalf ("Server with address %q not contacted: %v" , addrs [2 ], err )
1112
1112
}
1113
1113
1114
1114
// Second SubConn connects.
1115
- servers .enterReady (1 )
1115
+ servers [ 1 ] .enterReady ()
1116
1116
testutils .AwaitState (ctx , t , cc , connectivity .Connecting )
1117
1117
}
1118
1118
@@ -1221,37 +1221,27 @@ func (c *ccStateSubscriber) OnMessage(msg any) {
1221
1221
c .transitions = append (c .transitions , msg .(connectivity.State ))
1222
1222
}
1223
1223
1224
- // handingServerGroup is a group of servers that accept a TCP connection and
1225
- // remain idle until asked to close the connection. They can be used to control
1224
+ // hangingServer is a server that accept a TCP connection and remains idle until
1225
+ // asked to close or respond to the connection. They can be used to control
1226
1226
// how long it takes for a subchannel to report a TRANSIENT_FAILURE in tests.
1227
- type handingServerGroup struct {
1228
- addrs [] resolver.Address
1229
- listeners [] net.Listener
1230
- serverConnCloseFuncs [] func ()
1231
- serverContacted [] * grpcsync.Event
1232
- readyChans [] chan struct {}
1227
+ type hangingServer struct {
1228
+ addr resolver.Address
1229
+ listener net.Listener
1230
+ closeConn func ()
1231
+ contacted * grpcsync.Event
1232
+ readyChan chan struct {}
1233
1233
}
1234
1234
1235
- func newHangingServerGroup (t * testing.T , count int ) * handingServerGroup {
1236
- listeners := []net.Listener {}
1237
- closeFns := []func (){}
1238
- addrs := []resolver.Address {}
1239
- contactedEvents := []* grpcsync.Event {}
1240
- readyChans := []chan struct {}{}
1241
-
1235
+ func newHangingServerGroup (t * testing.T , count int ) []* hangingServer {
1236
+ servers := []* hangingServer {}
1242
1237
for i := 0 ; i < count ; i ++ {
1243
1238
lis , err := net .Listen ("tcp" , "localhost:0" )
1244
1239
if err != nil {
1245
1240
t .Fatalf ("Error while listening. Err: %v" , err )
1246
1241
}
1247
- listeners = append (listeners , lis )
1248
- addrs = append (addrs , resolver.Address {Addr : lis .Addr ().String ()})
1249
1242
closeChan := make (chan struct {})
1250
- closeFns = append (closeFns , sync .OnceFunc (func () { close (closeChan ) }))
1251
1243
contacted := grpcsync .NewEvent ()
1252
- contactedEvents = append (contactedEvents , contacted )
1253
1244
readyChan := make (chan struct {})
1254
- readyChans = append (readyChans , readyChan )
1255
1245
1256
1246
go func () {
1257
1247
conn , err := lis .Accept ()
@@ -1277,43 +1267,47 @@ func newHangingServerGroup(t *testing.T, count int) *handingServerGroup {
1277
1267
}
1278
1268
}
1279
1269
}()
1280
- }
1281
-
1282
- return & handingServerGroup {
1283
- addrs : addrs ,
1284
- listeners : listeners ,
1285
- serverConnCloseFuncs : closeFns ,
1286
- serverContacted : contactedEvents ,
1287
- readyChans : readyChans ,
1288
- }
1289
- }
1270
+ server := & hangingServer {
1271
+ addr : resolver. Address { Addr : lis . Addr (). String ()},
1272
+ listener : lis ,
1273
+ closeConn : grpcsync . OnceFunc ( func () {
1274
+ close ( closeChan )
1275
+ }) ,
1276
+ contacted : contacted ,
1277
+ readyChan : readyChan ,
1278
+ }
1279
+ servers = append ( servers , server )
1290
1280
1291
- func (hg * handingServerGroup ) close () {
1292
- for _ , fn := range hg .serverConnCloseFuncs {
1293
- fn ()
1294
- }
1295
- for _ , l := range hg .listeners {
1296
- l .Close ()
1281
+ t .Cleanup (func () {
1282
+ server .closeConn ()
1283
+ server .listener .Close ()
1284
+ })
1297
1285
}
1298
- }
1299
1286
1300
- func (hg * handingServerGroup ) closeConn (serverIdx int ) {
1301
- hg .serverConnCloseFuncs [serverIdx ]()
1287
+ return servers
1302
1288
}
1303
1289
1304
- func (hg * handingServerGroup ) isContacted (serverIdx int ) bool {
1305
- return hg . serverContacted [ serverIdx ] .HasFired ()
1290
+ func (s * hangingServer ) isContacted () bool {
1291
+ return s . contacted .HasFired ()
1306
1292
}
1307
1293
1308
- func (hg * handingServerGroup ) enterReady (serverIdx int ) {
1309
- hg . readyChans [ serverIdx ] <- struct {}{}
1294
+ func (s * hangingServer ) enterReady () {
1295
+ s . readyChan <- struct {}{}
1310
1296
}
1311
1297
1312
- func (hg * handingServerGroup ) awaitContacted (ctx context.Context , serverIdx int ) error {
1298
+ func (s * hangingServer ) awaitContacted (ctx context.Context ) error {
1313
1299
select {
1314
1300
case <- ctx .Done ():
1315
1301
return ctx .Err ()
1316
- case <- hg . serverContacted [ serverIdx ] .Done ():
1302
+ case <- s . contacted .Done ():
1317
1303
}
1318
1304
return nil
1319
1305
}
1306
+
1307
+ func resolverAddrsFromHangingServers (servers []* hangingServer ) []resolver.Address {
1308
+ addrs := []resolver.Address {}
1309
+ for _ , srv := range servers {
1310
+ addrs = append (addrs , srv .addr )
1311
+ }
1312
+ return addrs
1313
+ }
0 commit comments