mirror of https://github.com/grpc/grpc-go.git
301 lines
9.6 KiB
Go
301 lines
9.6 KiB
Go
/*
|
|
*
|
|
* Copyright 2025 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 xdsclient_test
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"google.golang.org/grpc/internal/grpctest"
|
|
"google.golang.org/grpc/xds/internal/clients/internal/pretty"
|
|
"google.golang.org/grpc/xds/internal/clients/internal/testutils"
|
|
"google.golang.org/grpc/xds/internal/clients/xdsclient"
|
|
"google.golang.org/grpc/xds/internal/clients/xdsclient/internal/xdsresource"
|
|
"google.golang.org/protobuf/proto"
|
|
|
|
v3listenerpb "github.com/envoyproxy/go-control-plane/envoy/config/listener/v3"
|
|
v3httppb "github.com/envoyproxy/go-control-plane/envoy/extensions/filters/network/http_connection_manager/v3"
|
|
"github.com/google/go-cmp/cmp"
|
|
)
|
|
|
|
type s struct {
|
|
grpctest.Tester
|
|
}
|
|
|
|
func Test(t *testing.T) {
|
|
grpctest.RunSubTests(t, s{})
|
|
}
|
|
|
|
const (
|
|
defaultTestWatchExpiryTimeout = 500 * time.Millisecond
|
|
defaultTestTimeout = 10 * time.Second
|
|
defaultTestShortTimeout = 10 * time.Millisecond // For events expected to *not* happen.
|
|
|
|
// ListenerResourceTypeName represents the transport agnostic name for the
|
|
// listener resource.
|
|
listenerResourceTypeName = "ListenerResource"
|
|
|
|
ldsName = "xdsclient-test-lds-resource"
|
|
rdsName = "xdsclient-test-rds-resource"
|
|
ldsNameNewStyle = "xdstp:///envoy.config.listener.v3.Listener/xdsclient-test-lds-resource"
|
|
rdsNameNewStyle = "xdstp:///envoy.config.route.v3.RouteConfiguration/xdsclient-test-rds-resource"
|
|
)
|
|
|
|
var (
|
|
// Singleton instantiation of the resource type implementation.
|
|
listenerType = xdsclient.ResourceType{
|
|
TypeURL: xdsresource.V3ListenerURL,
|
|
TypeName: listenerResourceTypeName,
|
|
AllResourcesRequiredInSotW: true,
|
|
Decoder: listenerDecoder{},
|
|
}
|
|
)
|
|
|
|
func unmarshalListenerResource(r []byte) (string, listenerUpdate, error) {
|
|
lis := &v3listenerpb.Listener{}
|
|
if err := proto.Unmarshal(r, lis); err != nil {
|
|
return "", listenerUpdate{}, fmt.Errorf("failed to unmarshal resource: %v", err)
|
|
}
|
|
|
|
lu, err := processListener(lis)
|
|
if err != nil {
|
|
return lis.GetName(), listenerUpdate{}, err
|
|
}
|
|
lu.Raw = r
|
|
return lis.GetName(), *lu, nil
|
|
}
|
|
|
|
func processListener(lis *v3listenerpb.Listener) (*listenerUpdate, error) {
|
|
if lis.GetApiListener() != nil {
|
|
return processClientSideListener(lis)
|
|
}
|
|
return processServerSideListener(lis)
|
|
}
|
|
|
|
// processClientSideListener checks if the provided Listener proto meets
|
|
// the expected criteria. If so, it returns a non-empty routeConfigName.
|
|
func processClientSideListener(lis *v3listenerpb.Listener) (*listenerUpdate, error) {
|
|
update := &listenerUpdate{}
|
|
|
|
apiLisAny := lis.GetApiListener().GetApiListener()
|
|
if !xdsresource.IsHTTPConnManagerResource(apiLisAny.GetTypeUrl()) {
|
|
return nil, fmt.Errorf("unexpected http connection manager resource type: %q", apiLisAny.GetTypeUrl())
|
|
}
|
|
apiLis := &v3httppb.HttpConnectionManager{}
|
|
if err := proto.Unmarshal(apiLisAny.GetValue(), apiLis); err != nil {
|
|
return nil, fmt.Errorf("failed to unmarshal api_listener: %v", err)
|
|
}
|
|
|
|
switch apiLis.RouteSpecifier.(type) {
|
|
case *v3httppb.HttpConnectionManager_Rds:
|
|
if configsource := apiLis.GetRds().GetConfigSource(); configsource.GetAds() == nil && configsource.GetSelf() == nil {
|
|
return nil, fmt.Errorf("LDS's RDS configSource is not ADS or Self: %+v", lis)
|
|
}
|
|
name := apiLis.GetRds().GetRouteConfigName()
|
|
if name == "" {
|
|
return nil, fmt.Errorf("empty route_config_name: %+v", lis)
|
|
}
|
|
update.RouteConfigName = name
|
|
case *v3httppb.HttpConnectionManager_RouteConfig:
|
|
routeU := apiLis.GetRouteConfig()
|
|
if routeU == nil {
|
|
return nil, fmt.Errorf("empty inline RDS resp:: %+v", lis)
|
|
}
|
|
if routeU.Name == "" {
|
|
return nil, fmt.Errorf("empty route_config_name in inline RDS resp: %+v", lis)
|
|
}
|
|
update.RouteConfigName = routeU.Name
|
|
case nil:
|
|
return nil, fmt.Errorf("no RouteSpecifier: %+v", apiLis)
|
|
default:
|
|
return nil, fmt.Errorf("unsupported type %T for RouteSpecifier", apiLis.RouteSpecifier)
|
|
}
|
|
|
|
return update, nil
|
|
}
|
|
|
|
func processServerSideListener(lis *v3listenerpb.Listener) (*listenerUpdate, error) {
|
|
if n := len(lis.ListenerFilters); n != 0 {
|
|
return nil, fmt.Errorf("unsupported field 'listener_filters' contains %d entries", n)
|
|
}
|
|
if lis.GetUseOriginalDst().GetValue() {
|
|
return nil, errors.New("unsupported field 'use_original_dst' is present and set to true")
|
|
}
|
|
addr := lis.GetAddress()
|
|
if addr == nil {
|
|
return nil, fmt.Errorf("no address field in LDS response: %+v", lis)
|
|
}
|
|
sockAddr := addr.GetSocketAddress()
|
|
if sockAddr == nil {
|
|
return nil, fmt.Errorf("no socket_address field in LDS response: %+v", lis)
|
|
}
|
|
lu := &listenerUpdate{
|
|
InboundListenerCfg: &inboundListenerConfig{
|
|
Address: sockAddr.GetAddress(),
|
|
Port: strconv.Itoa(int(sockAddr.GetPortValue())),
|
|
},
|
|
}
|
|
|
|
return lu, nil
|
|
}
|
|
|
|
type listenerDecoder struct{}
|
|
|
|
// Decode deserializes and validates an xDS resource serialized inside the
|
|
// provided `Any` proto, as received from the xDS management server.
|
|
func (listenerDecoder) Decode(resource []byte, _ xdsclient.DecodeOptions) (*xdsclient.DecodeResult, error) {
|
|
name, listener, err := unmarshalListenerResource(resource)
|
|
switch {
|
|
case name == "":
|
|
// Name is unset only when protobuf deserialization fails.
|
|
return nil, err
|
|
case err != nil:
|
|
// Protobuf deserialization succeeded, but resource validation failed.
|
|
return &xdsclient.DecodeResult{Name: name, Resource: &listenerResourceData{Resource: listenerUpdate{}}}, err
|
|
}
|
|
|
|
return &xdsclient.DecodeResult{Name: name, Resource: &listenerResourceData{Resource: listener}}, nil
|
|
|
|
}
|
|
|
|
// listenerResourceData wraps the configuration of a Listener resource as
|
|
// received from the management server.
|
|
//
|
|
// Implements the ResourceData interface.
|
|
type listenerResourceData struct {
|
|
xdsclient.ResourceData
|
|
|
|
Resource listenerUpdate
|
|
}
|
|
|
|
// Equal returns true if other is equal to l.
|
|
func (l *listenerResourceData) Equal(other xdsclient.ResourceData) bool {
|
|
if l == nil && other == nil {
|
|
return true
|
|
}
|
|
if (l == nil) != (other == nil) {
|
|
return false
|
|
}
|
|
return bytes.Equal(l.Resource.Raw, other.Bytes())
|
|
}
|
|
|
|
// ToJSON returns a JSON string representation of the resource data.
|
|
func (l *listenerResourceData) ToJSON() string {
|
|
return pretty.ToJSON(l.Resource)
|
|
}
|
|
|
|
// Bytes returns the underlying raw protobuf form of the listener resource.
|
|
func (l *listenerResourceData) Bytes() []byte {
|
|
return l.Resource.Raw
|
|
}
|
|
|
|
// ListenerUpdate contains information received in an LDS response, which is of
|
|
// interest to the registered LDS watcher.
|
|
type listenerUpdate struct {
|
|
// RouteConfigName is the route configuration name corresponding to the
|
|
// target which is being watched through LDS.
|
|
RouteConfigName string
|
|
|
|
// InboundListenerCfg contains inbound listener configuration.
|
|
InboundListenerCfg *inboundListenerConfig
|
|
|
|
// Raw is the resource from the xds response.
|
|
Raw []byte
|
|
}
|
|
|
|
// InboundListenerConfig contains information about the inbound listener, i.e
|
|
// the server-side listener.
|
|
type inboundListenerConfig struct {
|
|
// Address is the local address on which the inbound listener is expected to
|
|
// accept incoming connections.
|
|
Address string
|
|
// Port is the local port on which the inbound listener is expected to
|
|
// accept incoming connections.
|
|
Port string
|
|
}
|
|
|
|
func makeAuthorityName(name string) string {
|
|
segs := strings.Split(name, "/")
|
|
return strings.Join(segs, "")
|
|
}
|
|
|
|
func makeNewStyleLDSName(authority string) string {
|
|
return fmt.Sprintf("xdstp://%s/envoy.config.listener.v3.Listener/xdsclient-test-lds-resource", authority)
|
|
}
|
|
|
|
// buildResourceName returns the resource name in the format of an xdstp://
|
|
// resource.
|
|
func buildResourceName(typeName, auth, id string, ctxParams map[string]string) string {
|
|
var typS string
|
|
switch typeName {
|
|
case listenerResourceTypeName:
|
|
typS = "envoy.config.listener.v3.Listener"
|
|
default:
|
|
// If the name doesn't match any of the standard resources fallback
|
|
// to the type name.
|
|
typS = typeName
|
|
}
|
|
return (&xdsresource.Name{
|
|
Scheme: "xdstp",
|
|
Authority: auth,
|
|
Type: typS,
|
|
ID: id,
|
|
ContextParams: ctxParams,
|
|
}).String()
|
|
}
|
|
|
|
// testMetricsReporter is a MetricsReporter to be used in tests. It sends
|
|
// recording events on channels and provides helpers to check if certain events
|
|
// have taken place.
|
|
type testMetricsReporter struct {
|
|
metricsCh *testutils.Channel
|
|
}
|
|
|
|
// newTestMetricsReporter returns a new testMetricsReporter.
|
|
func newTestMetricsReporter() *testMetricsReporter {
|
|
return &testMetricsReporter{
|
|
metricsCh: testutils.NewChannelWithSize(1),
|
|
}
|
|
}
|
|
|
|
// waitForMetric waits for a metric to be recorded and verifies that the
|
|
// recorded metrics data matches the expected metricsDataWant. Returns
|
|
// an error if failed to wait or received wrong data.
|
|
func (r *testMetricsReporter) waitForMetric(ctx context.Context, metricsDataWant any) error {
|
|
got, err := r.metricsCh.Receive(ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("timeout waiting for int64Count")
|
|
}
|
|
if diff := cmp.Diff(got, metricsDataWant); diff != "" {
|
|
return fmt.Errorf("received unexpected metrics value (-got, +want): %v", diff)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ReportMetric sends the metrics data to the metricsCh channel.
|
|
func (r *testMetricsReporter) ReportMetric(m any) {
|
|
r.metricsCh.Replace(m)
|
|
}
|