@@ -22,16 +22,19 @@ import (
22
22
"context"
23
23
"encoding/json"
24
24
"fmt"
25
+ "strings"
25
26
"testing"
26
27
27
28
"github.com/google/uuid"
29
+ "google.golang.org/grpc"
28
30
"google.golang.org/grpc/internal/testutils"
29
31
"google.golang.org/grpc/internal/testutils/xds/e2e"
30
32
"google.golang.org/grpc/internal/testutils/xds/fakeserver"
31
33
"google.golang.org/grpc/internal/xds/bootstrap"
32
34
"google.golang.org/grpc/xds/internal"
33
35
xdstestutils "google.golang.org/grpc/xds/internal/testutils"
34
36
"google.golang.org/grpc/xds/internal/xdsclient"
37
+ xdsclientinternal "google.golang.org/grpc/xds/internal/xdsclient/internal"
35
38
"google.golang.org/grpc/xds/internal/xdsclient/xdsresource"
36
39
"google.golang.org/grpc/xds/internal/xdsclient/xdsresource/version"
37
40
"google.golang.org/protobuf/types/known/anypb"
@@ -371,3 +374,135 @@ func readDiscoveryResponseAndCheckForNonEmptyNodeProto(ctx context.Context, reqC
371
374
}
372
375
return nil
373
376
}
377
+
378
+ type testRouteConfigResourceType struct {}
379
+
380
+ func (testRouteConfigResourceType ) TypeURL () string { return version .V3RouteConfigURL }
381
+ func (testRouteConfigResourceType ) TypeName () string { return "RouteConfigResource" }
382
+ func (testRouteConfigResourceType ) AllResourcesRequiredInSotW () bool { return false }
383
+ func (testRouteConfigResourceType ) Decode (* xdsresource.DecodeOptions , * anypb.Any ) (* xdsresource.DecodeResult , error ) {
384
+ return nil , nil
385
+ }
386
+
387
+ // Tests that the errors returned by the xDS client when watching a resource
388
+ // contain the node ID that was used to create the client. This test covers two
389
+ // scenarios:
390
+ //
391
+ // 1. When a watch is registered for an already registered resource type, but
392
+ // this time with a different implementation,
393
+ // 2. When a watch is registered for a resource name whose authority is not
394
+ // found in the bootstrap configuration.
395
+ func (s ) TestWatchErrorsContainNodeID (t * testing.T ) {
396
+ mgmtServer := e2e .StartManagementServer (t , e2e.ManagementServerOptions {})
397
+
398
+ // Create bootstrap configuration pointing to the above management server.
399
+ nodeID := uuid .New ().String ()
400
+ bc := e2e .DefaultBootstrapContents (t , nodeID , mgmtServer .Address )
401
+
402
+ // Create an xDS client with the above bootstrap contents.
403
+ config , err := bootstrap .NewConfigFromContents (bc )
404
+ if err != nil {
405
+ t .Fatalf ("Failed to parse bootstrap contents: %s, %v" , string (bc ), err )
406
+ }
407
+ pool := xdsclient .NewPool (config )
408
+ client , close , err := pool .NewClientForTesting (xdsclient.OptionsForTesting {
409
+ Name : t .Name (),
410
+ })
411
+ if err != nil {
412
+ t .Fatalf ("Failed to create xDS client: %v" , err )
413
+ }
414
+ defer close ()
415
+
416
+ ctx , cancel := context .WithTimeout (context .Background (), defaultTestTimeout )
417
+ defer cancel ()
418
+
419
+ t .Run ("Multiple_ResourceType_Implementations" , func (t * testing.T ) {
420
+ const routeConfigName = "route-config-name"
421
+ watcher := xdstestutils .NewTestResourceWatcher ()
422
+ client .WatchResource (routeConfigResourceType , routeConfigName , watcher )
423
+
424
+ sCtx , sCancel := context .WithTimeout (ctx , defaultTestShortTimeout )
425
+ defer sCancel ()
426
+ select {
427
+ case <- sCtx .Done ():
428
+ case <- watcher .UpdateCh :
429
+ t .Fatal ("Unexpected resource update" )
430
+ case <- watcher .ErrorCh :
431
+ t .Fatal ("Unexpected resource error" )
432
+ case <- watcher .ResourceDoesNotExistCh :
433
+ t .Fatal ("Unexpected resource does not exist" )
434
+ }
435
+
436
+ client .WatchResource (testRouteConfigResourceType {}, routeConfigName , watcher )
437
+ select {
438
+ case <- ctx .Done ():
439
+ t .Fatal ("Timeout when waiting for error callback to be invoked" )
440
+ case err := <- watcher .ErrorCh :
441
+ if err == nil || ! strings .Contains (err .Error (), nodeID ) {
442
+ t .Fatalf ("Unexpected error: %v, want error with node ID: %q" , err , nodeID )
443
+ }
444
+ }
445
+ })
446
+
447
+ t .Run ("Missing_Authority" , func (t * testing.T ) {
448
+ const routeConfigName = "xdstp://nonexistant-authority/envoy.config.route.v3.RouteConfiguration/route-config-name"
449
+ watcher := xdstestutils .NewTestResourceWatcher ()
450
+ client .WatchResource (routeConfigResourceType , routeConfigName , watcher )
451
+
452
+ select {
453
+ case <- ctx .Done ():
454
+ t .Fatal ("Timeout when waiting for error callback to be invoked" )
455
+ case err := <- watcher .ErrorCh :
456
+ if err == nil || ! strings .Contains (err .Error (), nodeID ) {
457
+ t .Fatalf ("Unexpected error: %v, want error with node ID: %q" , err , nodeID )
458
+ }
459
+ }
460
+ })
461
+ }
462
+
463
+ // Tests that the errors returned by the xDS client when watching a resource
464
+ // contain the node ID when channel creation to the management server fails.
465
+ func (s ) TestWatchErrorsContainNodeID_ChannelCreationFailure (t * testing.T ) {
466
+ mgmtServer := e2e .StartManagementServer (t , e2e.ManagementServerOptions {})
467
+
468
+ // Create bootstrap configuration pointing to the above management server.
469
+ nodeID := uuid .New ().String ()
470
+ bc := e2e .DefaultBootstrapContents (t , nodeID , mgmtServer .Address )
471
+
472
+ // Create an xDS client with the above bootstrap contents.
473
+ config , err := bootstrap .NewConfigFromContents (bc )
474
+ if err != nil {
475
+ t .Fatalf ("Failed to parse bootstrap contents: %s, %v" , string (bc ), err )
476
+ }
477
+ pool := xdsclient .NewPool (config )
478
+ client , close , err := pool .NewClientForTesting (xdsclient.OptionsForTesting {
479
+ Name : t .Name (),
480
+ })
481
+ if err != nil {
482
+ t .Fatalf ("Failed to create xDS client: %v" , err )
483
+ }
484
+ defer close ()
485
+
486
+ ctx , cancel := context .WithTimeout (context .Background (), defaultTestTimeout )
487
+ defer cancel ()
488
+
489
+ // Override the xDS channel dialer with one that always fails.
490
+ origDialer := xdsclientinternal .GRPCNewClient
491
+ xdsclientinternal .GRPCNewClient = func (target string , opts ... grpc.DialOption ) (* grpc.ClientConn , error ) {
492
+ return nil , fmt .Errorf ("failed to create channel" )
493
+ }
494
+ defer func () { xdsclientinternal .GRPCNewClient = origDialer }()
495
+
496
+ const routeConfigName = "route-config-name"
497
+ watcher := xdstestutils .NewTestResourceWatcher ()
498
+ client .WatchResource (routeConfigResourceType , routeConfigName , watcher )
499
+
500
+ select {
501
+ case <- ctx .Done ():
502
+ t .Fatal ("Timeout when waiting for error callback to be invoked" )
503
+ case err := <- watcher .ErrorCh :
504
+ if err == nil || ! strings .Contains (err .Error (), nodeID ) {
505
+ t .Fatalf ("Unexpected error: %v, want error with node ID: %q" , err , nodeID )
506
+ }
507
+ }
508
+ }
0 commit comments