mirror of https://github.com/grpc/grpc-go.git
200 lines
6.3 KiB
Go
200 lines
6.3 KiB
Go
/*
|
|
*
|
|
* Copyright 2019 gRPC authors.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*/
|
|
|
|
package v2
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
|
|
v2xdspb "github.com/envoyproxy/go-control-plane/envoy/api/v2"
|
|
"github.com/google/go-cmp/cmp/cmpopts"
|
|
|
|
"google.golang.org/grpc/xds/internal/xdsclient"
|
|
)
|
|
|
|
// TestLDSHandleResponse starts a fake xDS server, makes a ClientConn to it,
|
|
// and creates a client using it. Then, it registers a watchLDS and tests
|
|
// different LDS responses.
|
|
func (s) TestLDSHandleResponse(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
ldsResponse *v2xdspb.DiscoveryResponse
|
|
wantErr bool
|
|
wantUpdate map[string]xdsclient.ListenerUpdateErrTuple
|
|
wantUpdateMD xdsclient.UpdateMetadata
|
|
wantUpdateErr bool
|
|
}{
|
|
// Badly marshaled LDS response.
|
|
{
|
|
name: "badly-marshaled-response",
|
|
ldsResponse: badlyMarshaledLDSResponse,
|
|
wantErr: true,
|
|
wantUpdate: nil,
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusNACKed,
|
|
ErrState: &xdsclient.UpdateErrorMetadata{
|
|
Err: cmpopts.AnyError,
|
|
},
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response does not contain Listener proto.
|
|
{
|
|
name: "no-listener-proto-in-response",
|
|
ldsResponse: badResourceTypeInLDSResponse,
|
|
wantErr: true,
|
|
wantUpdate: nil,
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusNACKed,
|
|
ErrState: &xdsclient.UpdateErrorMetadata{
|
|
Err: cmpopts.AnyError,
|
|
},
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// No APIListener in the response. Just one test case here for a bad
|
|
// ApiListener, since the others are covered in
|
|
// TestGetRouteConfigNameFromListener.
|
|
{
|
|
name: "no-apiListener-in-response",
|
|
ldsResponse: noAPIListenerLDSResponse,
|
|
wantErr: true,
|
|
wantUpdate: map[string]xdsclient.ListenerUpdateErrTuple{
|
|
goodLDSTarget1: {Err: cmpopts.AnyError},
|
|
},
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusNACKed,
|
|
ErrState: &xdsclient.UpdateErrorMetadata{
|
|
Err: cmpopts.AnyError,
|
|
},
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response contains one listener and it is good.
|
|
{
|
|
name: "one-good-listener",
|
|
ldsResponse: goodLDSResponse1,
|
|
wantErr: false,
|
|
wantUpdate: map[string]xdsclient.ListenerUpdateErrTuple{
|
|
goodLDSTarget1: {Update: xdsclient.ListenerUpdate{RouteConfigName: goodRouteName1, Raw: marshaledListener1}},
|
|
},
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusACKed,
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response contains multiple good listeners, including the one we are
|
|
// interested in.
|
|
{
|
|
name: "multiple-good-listener",
|
|
ldsResponse: ldsResponseWithMultipleResources,
|
|
wantErr: false,
|
|
wantUpdate: map[string]xdsclient.ListenerUpdateErrTuple{
|
|
goodLDSTarget1: {Update: xdsclient.ListenerUpdate{RouteConfigName: goodRouteName1, Raw: marshaledListener1}},
|
|
goodLDSTarget2: {Update: xdsclient.ListenerUpdate{RouteConfigName: goodRouteName1, Raw: marshaledListener2}},
|
|
},
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusACKed,
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response contains two good listeners (one interesting and one
|
|
// uninteresting), and one badly marshaled listener. This will cause a
|
|
// nack because the uninteresting listener will still be parsed.
|
|
{
|
|
name: "good-bad-ugly-listeners",
|
|
ldsResponse: goodBadUglyLDSResponse,
|
|
wantErr: true,
|
|
wantUpdate: map[string]xdsclient.ListenerUpdateErrTuple{
|
|
goodLDSTarget1: {Update: xdsclient.ListenerUpdate{RouteConfigName: goodRouteName1, Raw: marshaledListener1}},
|
|
goodLDSTarget2: {Err: cmpopts.AnyError},
|
|
},
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusNACKed,
|
|
ErrState: &xdsclient.UpdateErrorMetadata{
|
|
Err: cmpopts.AnyError,
|
|
},
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response contains one listener, but we are not interested in it.
|
|
{
|
|
name: "one-uninteresting-listener",
|
|
ldsResponse: goodLDSResponse2,
|
|
wantErr: false,
|
|
wantUpdate: map[string]xdsclient.ListenerUpdateErrTuple{
|
|
goodLDSTarget2: {Update: xdsclient.ListenerUpdate{RouteConfigName: goodRouteName1, Raw: marshaledListener2}},
|
|
},
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusACKed,
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
// Response constains no resources. This is the case where the server
|
|
// does not know about the target we are interested in.
|
|
{
|
|
name: "empty-response",
|
|
ldsResponse: emptyLDSResponse,
|
|
wantErr: false,
|
|
wantUpdate: nil,
|
|
wantUpdateMD: xdsclient.UpdateMetadata{
|
|
Status: xdsclient.ServiceStatusACKed,
|
|
},
|
|
wantUpdateErr: false,
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
testWatchHandle(t, &watchHandleTestcase{
|
|
rType: xdsclient.ListenerResource,
|
|
resourceName: goodLDSTarget1,
|
|
responseToHandle: test.ldsResponse,
|
|
wantHandleErr: test.wantErr,
|
|
wantUpdate: test.wantUpdate,
|
|
wantUpdateMD: test.wantUpdateMD,
|
|
wantUpdateErr: test.wantUpdateErr,
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestLDSHandleResponseWithoutWatch tests the case where the client receives
|
|
// an LDS response without a registered watcher.
|
|
func (s) TestLDSHandleResponseWithoutWatch(t *testing.T) {
|
|
_, cc, cleanup := startServerAndGetCC(t)
|
|
defer cleanup()
|
|
|
|
v2c, err := newV2Client(&testUpdateReceiver{
|
|
f: func(xdsclient.ResourceType, map[string]interface{}, xdsclient.UpdateMetadata) {},
|
|
}, cc, goodNodeProto, func(int) time.Duration { return 0 }, nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer v2c.Close()
|
|
|
|
if v2c.handleLDSResponse(badResourceTypeInLDSResponse) == nil {
|
|
t.Fatal("v2c.handleLDSResponse() succeeded, should have failed")
|
|
}
|
|
|
|
if v2c.handleLDSResponse(goodLDSResponse1) != nil {
|
|
t.Fatal("v2c.handleLDSResponse() succeeded, should have failed")
|
|
}
|
|
}
|