-
Notifications
You must be signed in to change notification settings - Fork 4.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
xdsclient: implementation of the xdsChannel (#7757)
- Loading branch information
Showing
9 changed files
with
1,296 additions
and
12 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,305 @@ | ||
/* | ||
* | ||
* Copyright 2024 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 | ||
|
||
import ( | ||
"errors" | ||
"fmt" | ||
"time" | ||
|
||
"google.golang.org/grpc/grpclog" | ||
"google.golang.org/grpc/internal/backoff" | ||
igrpclog "google.golang.org/grpc/internal/grpclog" | ||
"google.golang.org/grpc/internal/grpcsync" | ||
"google.golang.org/grpc/internal/xds/bootstrap" | ||
"google.golang.org/grpc/xds/internal/xdsclient/load" | ||
"google.golang.org/grpc/xds/internal/xdsclient/transport" | ||
"google.golang.org/grpc/xds/internal/xdsclient/transport/ads" | ||
"google.golang.org/grpc/xds/internal/xdsclient/transport/lrs" | ||
"google.golang.org/grpc/xds/internal/xdsclient/xdsresource" | ||
) | ||
|
||
// xdsChannelEventHandler wraps callbacks used to notify the xDS client about | ||
// events on the xdsChannel. Methods in this interface may be invoked | ||
// concurrently and the xDS client implementation needs to handle them in a | ||
// thread-safe manner. | ||
type xdsChannelEventHandler interface { | ||
// adsStreamFailure is called when the xdsChannel encounters an ADS stream | ||
// failure. | ||
adsStreamFailure(error) | ||
|
||
// adsResourceUpdate is called when the xdsChannel receives an ADS response | ||
// from the xDS management server. The callback is provided with the | ||
// following: | ||
// - the resource type of the resources in the response | ||
// - a map of resources in the response, keyed by resource name | ||
// - the metadata associated with the response | ||
// - a callback to be invoked when the updated is processed | ||
adsResourceUpdate(xdsresource.Type, map[string]ads.DataAndErrTuple, xdsresource.UpdateMetadata, func()) | ||
|
||
// adsResourceDoesNotExist is called when the xdsChannel determines that a | ||
// requested ADS resource does not exist. | ||
adsResourceDoesNotExist(xdsresource.Type, string) | ||
} | ||
|
||
// xdsChannelOpts holds the options for creating a new xdsChannel. | ||
type xdsChannelOpts struct { | ||
transport transport.Interface // Takes ownership of this transport. | ||
serverConfig *bootstrap.ServerConfig // Configuration of the server to connect to. | ||
bootstrapConfig *bootstrap.Config // Complete bootstrap configuration, used to decode resources. | ||
resourceTypeGetter func(string) xdsresource.Type // Function to retrieve resource parsing functionality, based on resource type. | ||
eventHandler xdsChannelEventHandler // Callbacks for ADS stream events. | ||
backoff func(int) time.Duration // Backoff function to use for stream retries. Defaults to exponential backoff, if unset. | ||
watchExpiryTimeout time.Duration // Timeout for ADS resource watch expiry. | ||
logPrefix string // Prefix to use for logging. | ||
} | ||
|
||
// newXDSChannel creates a new xdsChannel instance with the provided options. | ||
// It performs basic validation on the provided options and initializes the | ||
// xdsChannel with the necessary components. | ||
func newXDSChannel(opts xdsChannelOpts) (*xdsChannel, error) { | ||
switch { | ||
case opts.transport == nil: | ||
return nil, errors.New("xdsChannel: transport is nil") | ||
case opts.serverConfig == nil: | ||
return nil, errors.New("xdsChannel: serverConfig is nil") | ||
case opts.bootstrapConfig == nil: | ||
return nil, errors.New("xdsChannel: bootstrapConfig is nil") | ||
case opts.resourceTypeGetter == nil: | ||
return nil, errors.New("xdsChannel: resourceTypeGetter is nil") | ||
case opts.eventHandler == nil: | ||
return nil, errors.New("xdsChannel: eventHandler is nil") | ||
} | ||
|
||
xc := &xdsChannel{ | ||
transport: opts.transport, | ||
serverConfig: opts.serverConfig, | ||
bootstrapConfig: opts.bootstrapConfig, | ||
resourceTypeGetter: opts.resourceTypeGetter, | ||
eventHandler: opts.eventHandler, | ||
closed: grpcsync.NewEvent(), | ||
} | ||
|
||
l := grpclog.Component("xds") | ||
logPrefix := opts.logPrefix + fmt.Sprintf("[xds-channel %p] ", xc) | ||
xc.logger = igrpclog.NewPrefixLogger(l, logPrefix) | ||
|
||
if opts.backoff == nil { | ||
opts.backoff = backoff.DefaultExponential.Backoff | ||
} | ||
xc.ads = ads.NewStreamImpl(ads.StreamOpts{ | ||
Transport: xc.transport, | ||
EventHandler: xc, | ||
Backoff: opts.backoff, | ||
NodeProto: xc.bootstrapConfig.Node(), | ||
WatchExpiryTimeout: opts.watchExpiryTimeout, | ||
LogPrefix: logPrefix, | ||
}) | ||
xc.lrs = lrs.NewStreamImpl(lrs.StreamOpts{ | ||
Transport: xc.transport, | ||
Backoff: opts.backoff, | ||
NodeProto: xc.bootstrapConfig.Node(), | ||
LogPrefix: logPrefix, | ||
}) | ||
return xc, nil | ||
} | ||
|
||
// xdsChannel represents a client channel to a management server, and is | ||
// responsible for managing the lifecycle of the ADS and LRS streams. It invokes | ||
// callbacks on the registered event handler for various ADS stream events. | ||
type xdsChannel struct { | ||
// The following fields are initialized at creation time and are read-only | ||
// after that, and hence need not be guarded by a mutex. | ||
transport transport.Interface // Takes ownership of this transport (used to make streaming calls). | ||
ads *ads.StreamImpl // An ADS stream to the management server. | ||
lrs *lrs.StreamImpl // An LRS stream to the management server. | ||
serverConfig *bootstrap.ServerConfig // Configuration of the server to connect to. | ||
bootstrapConfig *bootstrap.Config // Complete bootstrap configuration, used to decode resources. | ||
resourceTypeGetter func(string) xdsresource.Type // Function to retrieve resource parsing functionality, based on resource type. | ||
eventHandler xdsChannelEventHandler // Callbacks for ADS stream events. | ||
logger *igrpclog.PrefixLogger // Logger to use for logging. | ||
closed *grpcsync.Event // Fired when the channel is closed. | ||
} | ||
|
||
func (xc *xdsChannel) close() { | ||
xc.closed.Fire() | ||
xc.ads.Stop() | ||
xc.lrs.Stop() | ||
xc.transport.Close() | ||
xc.logger.Infof("Shutdown") | ||
} | ||
|
||
// reportLoad returns a load.Store that can be used to report load to the LRS, and a | ||
// function that can be called to stop reporting load. | ||
func (xc *xdsChannel) reportLoad() (*load.Store, func()) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Attempt to start load reporting on closed channel") | ||
} | ||
return nil, func() {} | ||
} | ||
return xc.lrs.ReportLoad() | ||
} | ||
|
||
// subscribe adds a subscription for the given resource name of the given | ||
// resource type on the ADS stream. | ||
func (xc *xdsChannel) subscribe(typ xdsresource.Type, name string) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Attempt to subscribe to an xDS resource of type %s and name %q on a closed channel", typ.TypeName(), name) | ||
} | ||
return | ||
} | ||
xc.ads.Subscribe(typ, name) | ||
} | ||
|
||
// unsubscribe removes the subscription for the given resource name of the given | ||
// resource type from the ADS stream. | ||
func (xc *xdsChannel) unsubscribe(typ xdsresource.Type, name string) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Attempt to unsubscribe to an xDS resource of type %s and name %q on a closed channel", typ.TypeName(), name) | ||
} | ||
return | ||
} | ||
xc.ads.Unsubscribe(typ, name) | ||
} | ||
|
||
// The following OnADSXxx() methods implement the ads.StreamEventHandler interface | ||
// and are invoked by the ADS stream implementation. | ||
|
||
// OnADSStreamError is invoked when an error occurs on the ADS stream. It | ||
// propagates the update to the xDS client. | ||
func (xc *xdsChannel) OnADSStreamError(err error) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Received ADS stream error on a closed xdsChannel: %v", err) | ||
} | ||
return | ||
} | ||
xc.eventHandler.adsStreamFailure(err) | ||
} | ||
|
||
// OnADSWatchExpiry is invoked when a watch for a resource expires. It | ||
// propagates the update to the xDS client. | ||
func (xc *xdsChannel) OnADSWatchExpiry(typ xdsresource.Type, name string) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Received ADS resource watch expiry for resource %q on a closed xdsChannel", name) | ||
} | ||
return | ||
} | ||
xc.eventHandler.adsResourceDoesNotExist(typ, name) | ||
} | ||
|
||
// OnADSResponse is invoked when a response is received on the ADS stream. It | ||
// decodes the resources in the response, and propagates the updates to the xDS | ||
// client. | ||
// | ||
// It returns the list of resource names in the response and any errors | ||
// encountered during decoding. | ||
func (xc *xdsChannel) OnADSResponse(resp ads.Response, onDone func()) ([]string, error) { | ||
if xc.closed.HasFired() { | ||
if xc.logger.V(2) { | ||
xc.logger.Infof("Received an update from the ADS stream on closed ADS stream") | ||
} | ||
return nil, errors.New("xdsChannel is closed") | ||
} | ||
|
||
// Lookup the resource parser based on the resource type. | ||
rType := xc.resourceTypeGetter(resp.TypeURL) | ||
if rType == nil { | ||
return nil, xdsresource.NewErrorf(xdsresource.ErrorTypeResourceTypeUnsupported, "Resource type URL %q unknown in response from server", resp.TypeURL) | ||
} | ||
|
||
// Decode the resources and build the list of resource names to return. | ||
opts := &xdsresource.DecodeOptions{ | ||
BootstrapConfig: xc.bootstrapConfig, | ||
ServerConfig: xc.serverConfig, | ||
} | ||
updates, md, err := decodeResponse(opts, rType, resp) | ||
var names []string | ||
for name := range updates { | ||
names = append(names, name) | ||
} | ||
|
||
xc.eventHandler.adsResourceUpdate(rType, updates, md, onDone) | ||
return names, err | ||
} | ||
|
||
// decodeResponse decodes the resources in the given ADS response. | ||
// | ||
// The opts parameter provides configuration options for decoding the resources. | ||
// The rType parameter specifies the resource type parser to use for decoding | ||
// the resources. | ||
// | ||
// The returned map contains a key for each resource in the response, with the | ||
// value being either the decoded resource data or an error if decoding failed. | ||
// The returned metadata includes the version of the response, the timestamp of | ||
// the update, and the status of the update (ACKed or NACKed). | ||
// | ||
// If there are any errors decoding the resources, the metadata will indicate | ||
// that the update was NACKed, and the returned error will contain information | ||
// about all errors encountered by this function. | ||
func decodeResponse(opts *xdsresource.DecodeOptions, rType xdsresource.Type, resp ads.Response) (map[string]ads.DataAndErrTuple, xdsresource.UpdateMetadata, error) { | ||
timestamp := time.Now() | ||
md := xdsresource.UpdateMetadata{ | ||
Version: resp.Version, | ||
Timestamp: timestamp, | ||
} | ||
|
||
topLevelErrors := make([]error, 0) // Tracks deserialization errors, where we don't have a resource name. | ||
perResourceErrors := make(map[string]error) // Tracks resource validation errors, where we have a resource name. | ||
ret := make(map[string]ads.DataAndErrTuple) // Return result, a map from resource name to either resource data or error. | ||
for _, r := range resp.Resources { | ||
result, err := rType.Decode(opts, r) | ||
|
||
// Name field of the result is left unpopulated only when resource | ||
// deserialization fails. | ||
name := "" | ||
if result != nil { | ||
name = xdsresource.ParseName(result.Name).String() | ||
} | ||
if err == nil { | ||
ret[name] = ads.DataAndErrTuple{Resource: result.Resource} | ||
continue | ||
} | ||
if name == "" { | ||
topLevelErrors = append(topLevelErrors, err) | ||
continue | ||
} | ||
perResourceErrors[name] = err | ||
// Add place holder in the map so we know this resource name was in | ||
// the response. | ||
ret[name] = ads.DataAndErrTuple{Err: err} | ||
} | ||
|
||
if len(topLevelErrors) == 0 && len(perResourceErrors) == 0 { | ||
md.Status = xdsresource.ServiceStatusACKed | ||
return ret, md, nil | ||
} | ||
|
||
md.Status = xdsresource.ServiceStatusNACKed | ||
errRet := combineErrors(rType.TypeName(), topLevelErrors, perResourceErrors) | ||
md.ErrState = &xdsresource.UpdateErrorMetadata{ | ||
Version: resp.Version, | ||
Err: errRet, | ||
Timestamp: timestamp, | ||
} | ||
return ret, md, errRet | ||
} |
Oops, something went wrong.