connect: simplify the compiled discovery chain data structures (#6242)

This should make them better for sending over RPC or the API.

Instead of a chain implemented explicitly like a linked list (nodes
holding pointers to other nodes) instead switch to a flat map of named
nodes with nodes linking other other nodes by name. The shipped
structure is just a map and a string to indicate which key to start
from.

Other changes:

* inline the compiler option InferDefaults as true

* introduce compiled target config to avoid needing to send back
  additional maps of Resolvers; future target-specific compiled state
  can go here

* move compiled MeshGateway out of the Resolver and into the
  TargetConfig where it makes more sense.
pull/6246/head
R.B. Boyer 2019-08-01 22:44:05 -05:00 committed by GitHub
parent 3128937145
commit f02924fafe
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 1069 additions and 1051 deletions

View File

@ -361,7 +361,6 @@ func (c *ConfigEntry) ReadDiscoveryChain(args *structs.DiscoveryChainRequest, re
OverrideMeshGateway: args.OverrideMeshGateway, OverrideMeshGateway: args.OverrideMeshGateway,
OverrideProtocol: args.OverrideProtocol, OverrideProtocol: args.OverrideProtocol,
OverrideConnectTimeout: args.OverrideConnectTimeout, OverrideConnectTimeout: args.OverrideConnectTimeout,
InferDefaults: true,
Entries: entries, Entries: entries,
}) })
if err != nil { if err != nil {

View File

@ -33,8 +33,7 @@ type CompileRequest struct {
// overridden for any resolver in the compiled chain. // overridden for any resolver in the compiled chain.
OverrideConnectTimeout time.Duration OverrideConnectTimeout time.Duration
InferDefaults bool // TODO(rb): remove this? Entries *structs.DiscoveryChainConfigEntries
Entries *structs.DiscoveryChainConfigEntries
} }
// Compile assembles a discovery chain in the form of a graph of nodes using // Compile assembles a discovery chain in the form of a graph of nodes using
@ -56,7 +55,6 @@ func Compile(req CompileRequest) (*structs.CompiledDiscoveryChain, error) {
serviceName = req.ServiceName serviceName = req.ServiceName
currentNamespace = req.CurrentNamespace currentNamespace = req.CurrentNamespace
currentDatacenter = req.CurrentDatacenter currentDatacenter = req.CurrentDatacenter
inferDefaults = req.InferDefaults
entries = req.Entries entries = req.Entries
) )
if serviceName == "" { if serviceName == "" {
@ -79,15 +77,15 @@ func Compile(req CompileRequest) (*structs.CompiledDiscoveryChain, error) {
overrideMeshGateway: req.OverrideMeshGateway, overrideMeshGateway: req.OverrideMeshGateway,
overrideProtocol: req.OverrideProtocol, overrideProtocol: req.OverrideProtocol,
overrideConnectTimeout: req.OverrideConnectTimeout, overrideConnectTimeout: req.OverrideConnectTimeout,
inferDefaults: inferDefaults,
entries: entries, entries: entries,
splitterNodes: make(map[string]*structs.DiscoveryGraphNode), resolvers: make(map[string]*structs.ServiceResolverConfigEntry),
groupResolverNodes: make(map[structs.DiscoveryTarget]*structs.DiscoveryGraphNode), splitterNodes: make(map[string]*structs.DiscoveryGraphNode),
resolveNodes: make(map[structs.DiscoveryTarget]*structs.DiscoveryGraphNode),
resolvers: make(map[string]*structs.ServiceResolverConfigEntry), nodes: make(map[string]*structs.DiscoveryGraphNode),
retainResolvers: make(map[string]struct{}),
targets: make(map[structs.DiscoveryTarget]struct{}), targets: make(map[structs.DiscoveryTarget]structs.DiscoveryTargetConfig),
} }
if req.OverrideProtocol != "" { if req.OverrideProtocol != "" {
@ -113,16 +111,19 @@ type compiler struct {
overrideMeshGateway structs.MeshGatewayConfig overrideMeshGateway structs.MeshGatewayConfig
overrideProtocol string overrideProtocol string
overrideConnectTimeout time.Duration overrideConnectTimeout time.Duration
inferDefaults bool
// config entries that are being compiled (will be mutated during compilation) // config entries that are being compiled (will be mutated during compilation)
// //
// This is an INPUT field. // This is an INPUT field.
entries *structs.DiscoveryChainConfigEntries entries *structs.DiscoveryChainConfigEntries
// resolvers is initially seeded by copying the provided entries.Resolvers
// map and default resolvers are added as they are needed.
resolvers map[string]*structs.ServiceResolverConfigEntry
// cached nodes // cached nodes
splitterNodes map[string]*structs.DiscoveryGraphNode splitterNodes map[string]*structs.DiscoveryGraphNode
groupResolverNodes map[structs.DiscoveryTarget]*structs.DiscoveryGraphNode // this is also an OUTPUT field resolveNodes map[structs.DiscoveryTarget]*structs.DiscoveryGraphNode
// usesAdvancedRoutingFeatures is set to true if config entries for routing // usesAdvancedRoutingFeatures is set to true if config entries for routing
// or splitting appear in the compiled chain // or splitting appear in the compiled chain
@ -137,31 +138,24 @@ type compiler struct {
// This is an OUTPUT field. // This is an OUTPUT field.
customizedBy customizationMarkers customizedBy customizationMarkers
// topNode is computed inside of assembleChain()
//
// This is an OUTPUT field.
topNode *structs.DiscoveryGraphNode
// protocol is the common protocol used for all referenced services. These // protocol is the common protocol used for all referenced services. These
// cannot be mixed. // cannot be mixed.
// //
// This is an OUTPUT field. // This is an OUTPUT field.
protocol string protocol string
// resolvers is initially seeded by copying the provided entries.Resolvers // startNode is computed inside of assembleChain()
// map and default resolvers are added as they are needed.
//
// If redirects cause a resolver to not be needed it will be omitted from
// this map.
// //
// This is an OUTPUT field. // This is an OUTPUT field.
resolvers map[string]*structs.ServiceResolverConfigEntry startNode string
// retainResolvers flags the elements of the resolvers map that should be
// retained in the final results. // nodes is computed inside of compile()
retainResolvers map[string]struct{} //
// This is an OUTPUT field.
nodes map[string]*structs.DiscoveryGraphNode
// This is an OUTPUT field. // This is an OUTPUT field.
targets map[structs.DiscoveryTarget]struct{} targets map[structs.DiscoveryTarget]structs.DiscoveryTargetConfig
} }
type customizationMarkers struct { type customizationMarkers struct {
@ -174,6 +168,23 @@ func (m *customizationMarkers) IsZero() bool {
return !m.MeshGateway && !m.Protocol && !m.ConnectTimeout return !m.MeshGateway && !m.Protocol && !m.ConnectTimeout
} }
// recordNode stores the node internally in the compiled chain.
func (c *compiler) recordNode(node *structs.DiscoveryGraphNode) {
// Some types have their own type-specific lookups, so record those, too.
switch node.Type {
case structs.DiscoveryGraphNodeTypeRouter:
// no special storage
case structs.DiscoveryGraphNodeTypeSplitter:
c.splitterNodes[node.ServiceName()] = node
case structs.DiscoveryGraphNodeTypeResolver:
c.resolveNodes[node.Resolver.Target] = node
default:
panic("unknown node type '" + node.Type + "'")
}
c.nodes[node.MapKey()] = node
}
func (c *compiler) recordServiceProtocol(serviceName string) error { func (c *compiler) recordServiceProtocol(serviceName string) error {
if serviceDefault := c.entries.GetService(serviceName); serviceDefault != nil { if serviceDefault := c.entries.GetService(serviceName); serviceDefault != nil {
return c.recordProtocol(serviceName, serviceDefault.Protocol) return c.recordProtocol(serviceName, serviceDefault.Protocol)
@ -220,11 +231,12 @@ func (c *compiler) compile() (*structs.CompiledDiscoveryChain, error) {
return nil, err return nil, err
} }
if c.topNode == nil { // We don't need these intermediates anymore.
if c.inferDefaults { c.splitterNodes = nil
panic("impossible to return no results with infer defaults set to true") c.resolveNodes = nil
}
return nil, nil if c.startNode == "" {
panic("impossible to return no results")
} }
if err := c.detectCircularSplits(); err != nil { if err := c.detectCircularSplits(); err != nil {
@ -236,11 +248,8 @@ func (c *compiler) compile() (*structs.CompiledDiscoveryChain, error) {
c.flattenAdjacentSplitterNodes() c.flattenAdjacentSplitterNodes()
// Remove any unused resolvers. if err := c.removeUnusedNodes(); err != nil {
for name, _ := range c.resolvers { return nil, err
if _, ok := c.retainResolvers[name]; !ok {
delete(c.resolvers, name)
}
} }
if !enableAdvancedRoutingForProtocol(c.protocol) && c.usesAdvancedRoutingFeatures { if !enableAdvancedRoutingForProtocol(c.protocol) && c.usesAdvancedRoutingFeatures {
@ -252,12 +261,6 @@ func (c *compiler) compile() (*structs.CompiledDiscoveryChain, error) {
} }
} }
targets := make([]structs.DiscoveryTarget, 0, len(c.targets))
for target, _ := range c.targets {
targets = append(targets, target)
}
structs.DiscoveryTargets(targets).Sort()
if c.overrideProtocol != "" { if c.overrideProtocol != "" {
if c.overrideProtocol != c.protocol { if c.overrideProtocol != c.protocol {
c.protocol = c.overrideProtocol c.protocol = c.overrideProtocol
@ -290,15 +293,14 @@ func (c *compiler) compile() (*structs.CompiledDiscoveryChain, error) {
} }
return &structs.CompiledDiscoveryChain{ return &structs.CompiledDiscoveryChain{
ServiceName: c.serviceName, ServiceName: c.serviceName,
Namespace: c.currentNamespace, Namespace: c.currentNamespace,
Datacenter: c.currentDatacenter, Datacenter: c.currentDatacenter,
CustomizationHash: customizationHash, CustomizationHash: customizationHash,
Protocol: c.protocol, Protocol: c.protocol,
Node: c.topNode, StartNode: c.startNode,
Resolvers: c.resolvers, Nodes: c.nodes,
Targets: targets, Targets: c.targets,
GroupResolverNodes: c.groupResolverNodes, // TODO(rb): prune unused
}, nil }, nil
} }
@ -315,23 +317,28 @@ func (c *compiler) detectCircularResolves() error {
func (c *compiler) flattenAdjacentSplitterNodes() { func (c *compiler) flattenAdjacentSplitterNodes() {
for { for {
anyChanged := false anyChanged := false
for _, splitterNode := range c.splitterNodes { for _, node := range c.nodes {
fixedSplits := make([]*structs.DiscoverySplit, 0, len(splitterNode.Splits)) if node.Type != structs.DiscoveryGraphNodeTypeSplitter {
continue
}
fixedSplits := make([]*structs.DiscoverySplit, 0, len(node.Splits))
changed := false changed := false
for _, split := range splitterNode.Splits { for _, split := range node.Splits {
if split.Node.Type != structs.DiscoveryGraphNodeTypeSplitter { nextNode := c.nodes[split.NextNode]
if nextNode.Type != structs.DiscoveryGraphNodeTypeSplitter {
fixedSplits = append(fixedSplits, split) fixedSplits = append(fixedSplits, split)
continue continue
} }
changed = true changed = true
for _, innerSplit := range split.Node.Splits { for _, innerSplit := range nextNode.Splits {
effectiveWeight := split.Weight * innerSplit.Weight / 100 effectiveWeight := split.Weight * innerSplit.Weight / 100
newDiscoverySplit := &structs.DiscoverySplit{ newDiscoverySplit := &structs.DiscoverySplit{
Weight: structs.NormalizeServiceSplitWeight(effectiveWeight), Weight: structs.NormalizeServiceSplitWeight(effectiveWeight),
Node: innerSplit.Node, NextNode: innerSplit.NextNode,
} }
fixedSplits = append(fixedSplits, newDiscoverySplit) fixedSplits = append(fixedSplits, newDiscoverySplit)
@ -339,7 +346,7 @@ func (c *compiler) flattenAdjacentSplitterNodes() {
} }
if changed { if changed {
splitterNode.Splits = fixedSplits node.Splits = fixedSplits
anyChanged = true anyChanged = true
} }
} }
@ -350,21 +357,81 @@ func (c *compiler) flattenAdjacentSplitterNodes() {
} }
} }
// removeUnusedNodes walks the chain from the start and prunes any nodes that
// are no longer referenced. This can happen as a result of operations like
// flattenAdjacentSplitterNodes().
func (c *compiler) removeUnusedNodes() error {
var (
visited = make(map[string]struct{})
todo = make(map[string]struct{})
)
todo[c.startNode] = struct{}{}
getNext := func() string {
if len(todo) == 0 {
return ""
}
for k, _ := range todo {
delete(todo, k)
return k
}
return ""
}
for {
next := getNext()
if next == "" {
break
}
if _, ok := visited[next]; ok {
continue
}
visited[next] = struct{}{}
node := c.nodes[next]
if node == nil {
return fmt.Errorf("compilation references non-retained node %q", next)
}
switch node.Type {
case structs.DiscoveryGraphNodeTypeRouter:
for _, route := range node.Routes {
todo[route.NextNode] = struct{}{}
}
case structs.DiscoveryGraphNodeTypeSplitter:
for _, split := range node.Splits {
todo[split.NextNode] = struct{}{}
}
case structs.DiscoveryGraphNodeTypeResolver:
// nothing special
default:
return fmt.Errorf("unknown node type %q", node.Type)
}
}
if len(visited) == len(c.nodes) {
return nil
}
for name, _ := range c.nodes {
if _, ok := visited[name]; !ok {
delete(c.nodes, name)
}
}
return nil
}
// assembleChain will do the initial assembly of a chain of DiscoveryGraphNode // assembleChain will do the initial assembly of a chain of DiscoveryGraphNode
// entries from the provided config entries. No default resolvers are injected // entries from the provided config entries.
// here so it is expected that if there are no discovery chain config entries
// set up for a given service that it will produce no topNode from this.
func (c *compiler) assembleChain() error { func (c *compiler) assembleChain() error {
if c.topNode != nil { if c.startNode != "" || len(c.nodes) > 0 {
return fmt.Errorf("assembleChain should only be called once") return fmt.Errorf("assembleChain should only be called once")
} }
// Check for short circuit path. // Check for short circuit path.
if len(c.resolvers) == 0 && c.entries.IsChainEmpty() { if len(c.resolvers) == 0 && c.entries.IsChainEmpty() {
if !c.inferDefaults {
return nil // nothing explicitly configured
}
// Materialize defaults and cache. // Materialize defaults and cache.
c.resolvers[c.serviceName] = newDefaultServiceResolver(c.serviceName) c.resolvers[c.serviceName] = newDefaultServiceResolver(c.serviceName)
} }
@ -380,12 +447,12 @@ func (c *compiler) assembleChain() error {
if router == nil { if router == nil {
// If no router is configured, move on down the line to the next hop of // If no router is configured, move on down the line to the next hop of
// the chain. // the chain.
node, err := c.getSplitterOrGroupResolverNode(c.newTarget(c.serviceName, "", "", "")) node, err := c.getSplitterOrResolverNode(c.newTarget(c.serviceName, "", "", ""))
if err != nil { if err != nil {
return err return err
} }
c.topNode = node c.startNode = node.MapKey()
return nil return nil
} }
@ -419,11 +486,11 @@ func (c *compiler) assembleChain() error {
err error err error
) )
if dest.ServiceSubset == "" && dest.Namespace == "" { if dest.ServiceSubset == "" && dest.Namespace == "" {
node, err = c.getSplitterOrGroupResolverNode( node, err = c.getSplitterOrResolverNode(
c.newTarget(svc, dest.ServiceSubset, dest.Namespace, ""), c.newTarget(svc, dest.ServiceSubset, dest.Namespace, ""),
) )
} else { } else {
node, err = c.getGroupResolverNode( node, err = c.getResolverNode(
c.newTarget(svc, dest.ServiceSubset, dest.Namespace, ""), c.newTarget(svc, dest.ServiceSubset, dest.Namespace, ""),
false, false,
) )
@ -431,23 +498,25 @@ func (c *compiler) assembleChain() error {
if err != nil { if err != nil {
return err return err
} }
compiledRoute.DestinationNode = node compiledRoute.NextNode = node.MapKey()
} }
// If we have a router, we'll add a catch-all route at the end to send // If we have a router, we'll add a catch-all route at the end to send
// unmatched traffic to the next hop in the chain. // unmatched traffic to the next hop in the chain.
defaultDestinationNode, err := c.getSplitterOrGroupResolverNode(c.newTarget(c.serviceName, "", "", "")) defaultDestinationNode, err := c.getSplitterOrResolverNode(c.newTarget(c.serviceName, "", "", ""))
if err != nil { if err != nil {
return err return err
} }
defaultRoute := &structs.DiscoveryRoute{ defaultRoute := &structs.DiscoveryRoute{
Definition: newDefaultServiceRoute(c.serviceName), Definition: newDefaultServiceRoute(c.serviceName),
DestinationNode: defaultDestinationNode, NextNode: defaultDestinationNode.MapKey(),
} }
routeNode.Routes = append(routeNode.Routes, defaultRoute) routeNode.Routes = append(routeNode.Routes, defaultRoute)
c.topNode = routeNode c.startNode = routeNode.MapKey()
c.recordNode(routeNode)
return nil return nil
} }
@ -476,18 +545,17 @@ func (c *compiler) newTarget(service, serviceSubset, namespace, datacenter strin
} }
} }
func (c *compiler) getSplitterOrGroupResolverNode(target structs.DiscoveryTarget) (*structs.DiscoveryGraphNode, error) { func (c *compiler) getSplitterOrResolverNode(target structs.DiscoveryTarget) (*structs.DiscoveryGraphNode, error) {
nextNode, err := c.getSplitterNode(target.Service) nextNode, err := c.getSplitterNode(target.Service)
if err != nil { if err != nil {
return nil, err return nil, err
} else if nextNode != nil { } else if nextNode != nil {
return nextNode, nil return nextNode, nil
} }
return c.getGroupResolverNode(target, false) return c.getResolverNode(target, false)
} }
func (c *compiler) getSplitterNode(name string) (*structs.DiscoveryGraphNode, error) { func (c *compiler) getSplitterNode(name string) (*structs.DiscoveryGraphNode, error) {
// Do we already have the node? // Do we already have the node?
if prev, ok := c.splitterNodes[name]; ok { if prev, ok := c.splitterNodes[name]; ok {
return prev, nil return prev, nil
@ -512,7 +580,7 @@ func (c *compiler) getSplitterNode(name string) (*structs.DiscoveryGraphNode, er
// If we record this exists before recursing down it will short-circuit // If we record this exists before recursing down it will short-circuit
// sanely if there is some sort of graph loop below. // sanely if there is some sort of graph loop below.
c.splitterNodes[name] = splitNode c.recordNode(splitNode)
for _, split := range splitter.Splits { for _, split := range splitter.Splits {
compiledSplit := &structs.DiscoverySplit{ compiledSplit := &structs.DiscoverySplit{
@ -527,33 +595,33 @@ func (c *compiler) getSplitterNode(name string) (*structs.DiscoveryGraphNode, er
if err != nil { if err != nil {
return nil, err return nil, err
} else if nextNode != nil { } else if nextNode != nil {
compiledSplit.Node = nextNode compiledSplit.NextNode = nextNode.MapKey()
continue continue
} }
// fall through to group-resolver // fall through to group-resolver
} }
node, err := c.getGroupResolverNode( node, err := c.getResolverNode(
c.newTarget(svc, split.ServiceSubset, split.Namespace, ""), c.newTarget(svc, split.ServiceSubset, split.Namespace, ""),
false, false,
) )
if err != nil { if err != nil {
return nil, err return nil, err
} }
compiledSplit.Node = node compiledSplit.NextNode = node.MapKey()
} }
c.usesAdvancedRoutingFeatures = true c.usesAdvancedRoutingFeatures = true
return splitNode, nil return splitNode, nil
} }
// getGroupResolverNode handles most of the code to handle // getResolverNode handles most of the code to handle redirection/rewriting
// redirection/rewriting capabilities from a resolver config entry. It recurses // capabilities from a resolver config entry. It recurses into itself to
// into itself to _generate_ targets used for failover out of convenience. // _generate_ targets used for failover out of convenience.
func (c *compiler) getGroupResolverNode(target structs.DiscoveryTarget, recursedForFailover bool) (*structs.DiscoveryGraphNode, error) { func (c *compiler) getResolverNode(target structs.DiscoveryTarget, recursedForFailover bool) (*structs.DiscoveryGraphNode, error) {
RESOLVE_AGAIN: RESOLVE_AGAIN:
// Do we already have the node? // Do we already have the node?
if prev, ok := c.groupResolverNodes[target]; ok { if prev, ok := c.resolveNodes[target]; ok {
return prev, nil return prev, nil
} }
@ -603,9 +671,6 @@ RESOLVE_AGAIN:
} }
} }
// Since we're actually building a node with it, we can keep it.
c.retainResolvers[target.Service] = struct{}{}
connectTimeout := resolver.ConnectTimeout connectTimeout := resolver.ConnectTimeout
if connectTimeout < 1 { if connectTimeout < 1 {
connectTimeout = 5 * time.Second connectTimeout = 5 * time.Second
@ -619,48 +684,51 @@ RESOLVE_AGAIN:
} }
// Build node. // Build node.
groupResolverNode := &structs.DiscoveryGraphNode{ node := &structs.DiscoveryGraphNode{
Type: structs.DiscoveryGraphNodeTypeGroupResolver, Type: structs.DiscoveryGraphNodeTypeResolver,
Name: resolver.Name, Name: target.Identifier(),
GroupResolver: &structs.DiscoveryGroupResolver{ Resolver: &structs.DiscoveryResolver{
Definition: resolver, Definition: resolver,
Default: resolver.IsDefault(), Default: resolver.IsDefault(),
Target: target, Target: target,
ConnectTimeout: connectTimeout, ConnectTimeout: connectTimeout,
}, },
} }
groupResolver := groupResolverNode.GroupResolver
// Default mesh gateway settings targetConfig := structs.DiscoveryTargetConfig{
if serviceDefault := c.entries.GetService(resolver.Name); serviceDefault != nil { Subset: resolver.Subsets[target.ServiceSubset],
groupResolver.MeshGateway = serviceDefault.MeshGateway
} }
if c.entries.GlobalProxy != nil && groupResolver.MeshGateway.Mode == structs.MeshGatewayModeDefault { // Default mesh gateway settings
groupResolver.MeshGateway.Mode = c.entries.GlobalProxy.MeshGateway.Mode if serviceDefault := c.entries.GetService(target.Service); serviceDefault != nil {
targetConfig.MeshGateway = serviceDefault.MeshGateway
}
if c.entries.GlobalProxy != nil && targetConfig.MeshGateway.Mode == structs.MeshGatewayModeDefault {
targetConfig.MeshGateway.Mode = c.entries.GlobalProxy.MeshGateway.Mode
} }
if c.overrideMeshGateway.Mode != structs.MeshGatewayModeDefault { if c.overrideMeshGateway.Mode != structs.MeshGatewayModeDefault {
if groupResolver.MeshGateway.Mode != c.overrideMeshGateway.Mode { if targetConfig.MeshGateway.Mode != c.overrideMeshGateway.Mode {
groupResolver.MeshGateway.Mode = c.overrideMeshGateway.Mode targetConfig.MeshGateway.Mode = c.overrideMeshGateway.Mode
c.customizedBy.MeshGateway = true c.customizedBy.MeshGateway = true
} }
} }
// Retain this target even if we may not retain the group resolver. // Retain this target even if we may not retain the group resolver.
c.targets[target] = struct{}{} c.targets[target] = targetConfig
if recursedForFailover { if recursedForFailover {
// If we recursed here from ourselves in a failover context, just emit // If we recursed here from ourselves in a failover context, just emit
// this node without caching it or even processing failover again. // this node without caching it or even processing failover again.
// This is a little weird but it keeps the redirect/default-subset // This is a little weird but it keeps the redirect/default-subset
// logic in one place. // logic in one place.
return groupResolverNode, nil return node, nil
} }
// If we record this exists before recursing down it will short-circuit // If we record this exists before recursing down it will short-circuit
// sanely if there is some sort of graph loop below. // sanely if there is some sort of graph loop below.
c.groupResolverNodes[target] = groupResolverNode c.recordNode(node)
if len(resolver.Failover) > 0 { if len(resolver.Failover) > 0 {
f := resolver.Failover f := resolver.Failover
@ -705,23 +773,23 @@ RESOLVE_AGAIN:
df := &structs.DiscoveryFailover{ df := &structs.DiscoveryFailover{
Definition: &failover, Definition: &failover,
} }
groupResolver.Failover = df node.Resolver.Failover = df
// Convert the targets into targets by cheating a bit and // Convert the targets into targets by cheating a bit and
// recursing into ourselves. // recursing into ourselves.
for _, target := range failoverTargets { for _, target := range failoverTargets {
failoverGroupResolverNode, err := c.getGroupResolverNode(target, true) failoverResolveNode, err := c.getResolverNode(target, true)
if err != nil { if err != nil {
return nil, err return nil, err
} }
failoverTarget := failoverGroupResolverNode.GroupResolver.Target failoverTarget := failoverResolveNode.Resolver.Target
df.Targets = append(df.Targets, failoverTarget) df.Targets = append(df.Targets, failoverTarget)
} }
} }
} }
} }
return groupResolverNode, nil return node, nil
} }
func newDefaultServiceResolver(serviceName string) *structs.ServiceResolverConfigEntry { func newDefaultServiceResolver(serviceName string) *structs.ServiceResolverConfigEntry {

File diff suppressed because it is too large Load Diff

View File

@ -22,7 +22,6 @@ func TestCompileConfigEntries(
ServiceName: serviceName, ServiceName: serviceName,
CurrentNamespace: currentNamespace, CurrentNamespace: currentNamespace,
CurrentDatacenter: currentDatacenter, CurrentDatacenter: currentDatacenter,
InferDefaults: true,
Entries: set, Entries: set,
} }
if setup != nil { if setup != nil {

View File

@ -559,37 +559,6 @@ func (s *state) resetWatchesFromChain(
return fmt.Errorf("not possible to arrive here with no discovery chain") return fmt.Errorf("not possible to arrive here with no discovery chain")
} }
// Collect all sorts of catalog queries we'll have to run.
targets := make(map[structs.DiscoveryTarget]*structs.ServiceResolverConfigEntry)
addTarget := func(target structs.DiscoveryTarget) error {
resolver, ok := chain.Resolvers[target.Service]
if !ok {
return fmt.Errorf("missing resolver %q for target %s", target.Service, target)
}
targets[target] = resolver
return nil
}
// NOTE: We will NEVER see a missing chain, because we always request it with defaulting enabled.
meshGatewayModes := make(map[structs.DiscoveryTarget]structs.MeshGatewayMode)
for _, group := range chain.GroupResolverNodes {
groupResolver := group.GroupResolver
meshGatewayModes[groupResolver.Target] = groupResolver.MeshGateway.Mode
if err := addTarget(groupResolver.Target); err != nil {
return err
}
if groupResolver.Failover != nil {
for _, target := range groupResolver.Failover.Targets {
if err := addTarget(target); err != nil {
return err
}
}
}
}
// Initialize relevant sub maps. // Initialize relevant sub maps.
if _, ok := snap.ConnectProxy.WatchedUpstreams[id]; !ok { if _, ok := snap.ConnectProxy.WatchedUpstreams[id]; !ok {
snap.ConnectProxy.WatchedUpstreams[id] = make(map[structs.DiscoveryTarget]context.CancelFunc) snap.ConnectProxy.WatchedUpstreams[id] = make(map[structs.DiscoveryTarget]context.CancelFunc)
@ -611,35 +580,17 @@ func (s *state) resetWatchesFromChain(
cancelFn() cancelFn()
} }
for target, resolver := range targets { for target, targetConfig := range chain.Targets {
if target.Service != resolver.Name {
panic(target.Service + " != " + resolver.Name) // TODO(rb): remove
}
s.logger.Printf("[TRACE] proxycfg: upstream=%q:chain=%q: initializing watch of target %s", id, chain.ServiceName, target) s.logger.Printf("[TRACE] proxycfg: upstream=%q:chain=%q: initializing watch of target %s", id, chain.ServiceName, target)
// TODO(rb): make sure the cross-dc request properly fills in the alternate datacenters encodedTarget := target.Identifier()
var subset structs.ServiceResolverSubset
if target.ServiceSubset != "" {
var ok bool
subset, ok = resolver.Subsets[target.ServiceSubset]
if !ok {
// Not possible really.
return fmt.Errorf("target %s cannot be resolved; service %q does not have a subset named %q", target, target.Service, target.ServiceSubset)
}
}
encodedTarget, err := target.MarshalText()
if err != nil {
return fmt.Errorf("target %s cannot be converted into a cache key string: %v", target, err)
}
ctx, cancel := context.WithCancel(s.ctx) ctx, cancel := context.WithCancel(s.ctx)
// TODO (mesh-gateway)- maybe allow using a gateway within a datacenter at some point // TODO (mesh-gateway)- maybe allow using a gateway within a datacenter at some point
meshGateway := structs.MeshGatewayModeDefault meshGateway := structs.MeshGatewayModeDefault
if target.Datacenter != s.source.Datacenter { if target.Datacenter != s.source.Datacenter {
meshGateway = meshGatewayModes[target] meshGateway = targetConfig.MeshGateway.Mode
} }
// if the default mode // if the default mode
@ -647,12 +598,12 @@ func (s *state) resetWatchesFromChain(
meshGateway = structs.MeshGatewayModeNone meshGateway = structs.MeshGatewayModeNone
} }
err = s.watchConnectProxyService( err := s.watchConnectProxyService(
ctx, ctx,
"upstream-target:"+string(encodedTarget)+":"+id, "upstream-target:"+encodedTarget+":"+id,
target.Service, target.Service,
target.Datacenter, target.Datacenter,
subset.Filter, targetConfig.Subset.Filter,
meshGateway, meshGateway,
) )
if err != nil { if err != nil {

View File

@ -5,7 +5,6 @@ import (
"encoding" "encoding"
"fmt" "fmt"
"net/url" "net/url"
"sort"
"strings" "strings"
"time" "time"
) )
@ -30,24 +29,19 @@ type CompiledDiscoveryChain struct {
// Protocol is the overall protocol shared by everything in the chain. // Protocol is the overall protocol shared by everything in the chain.
Protocol string Protocol string
// Node is the top node in the chain. // StartNode is the first key into the Nodes map that should be followed
// // when walking the discovery chain.
// If this is a router or splitter then in envoy this renders as an http StartNode string `json:",omitempty"`
// route object.
//
// If this is a group resolver then in envoy this renders as a default
// wildcard http route object.
Node *DiscoveryGraphNode `json:",omitempty"`
// GroupResolverNodes respresents all unique service instance groups that // Nodes contains all nodes available for traversal in the chain keyed by a
// need to be represented. For envoy these render as Clusters. // unique name. You can walk this by starting with StartNode.
// //
// Omitted from JSON because these already show up under the Node field. // NOTE: The names should be treated as opaque values and are only
GroupResolverNodes map[DiscoveryTarget]*DiscoveryGraphNode `json:"-"` // guaranteed to be consistent within a single compilation.
Nodes map[string]*DiscoveryGraphNode `json:",omitempty"`
// TODO(rb): not sure if these two fields are actually necessary but I'll know when I get into xDS // Targets is a list of all targets and configuration related just to targets.
Resolvers map[string]*ServiceResolverConfigEntry `json:",omitempty"` Targets map[DiscoveryTarget]DiscoveryTargetConfig `json:",omitempty"`
Targets []DiscoveryTarget `json:",omitempty"`
} }
// IsDefault returns true if the compiled chain represents no routing, no // IsDefault returns true if the compiled chain represents no routing, no
@ -56,41 +50,31 @@ type CompiledDiscoveryChain struct {
// applied is redirection to another resolver that is default, so we double // applied is redirection to another resolver that is default, so we double
// check the resolver matches the requested resolver. // check the resolver matches the requested resolver.
func (c *CompiledDiscoveryChain) IsDefault() bool { func (c *CompiledDiscoveryChain) IsDefault() bool {
if c.Node == nil { if c.StartNode == "" || len(c.Nodes) == 0 {
return true return true
} }
node := c.Nodes[c.StartNode]
if node == nil {
panic("not possible: missing node named '" + c.StartNode + "' in chain '" + c.ServiceName + "'")
}
// TODO(rb): include CustomizationHash here? // TODO(rb): include CustomizationHash here?
return c.Node.Name == c.ServiceName && return node.Type == DiscoveryGraphNodeTypeResolver &&
c.Node.Type == DiscoveryGraphNodeTypeGroupResolver && node.Resolver.Default &&
c.Node.GroupResolver.Default node.Resolver.Target.Service == c.ServiceName
}
// SubsetDefinitionForTarget is a convenience function to fetch the subset
// definition for the service subset defined by the provided target. If the
// subset is not defined an empty definition is returned.
func (c *CompiledDiscoveryChain) SubsetDefinitionForTarget(t DiscoveryTarget) ServiceResolverSubset {
if t.ServiceSubset == "" {
return ServiceResolverSubset{}
}
resolver, ok := c.Resolvers[t.Service]
if !ok {
return ServiceResolverSubset{}
}
return resolver.Subsets[t.ServiceSubset]
} }
const ( const (
DiscoveryGraphNodeTypeRouter = "router" DiscoveryGraphNodeTypeRouter = "router"
DiscoveryGraphNodeTypeSplitter = "splitter" DiscoveryGraphNodeTypeSplitter = "splitter"
DiscoveryGraphNodeTypeGroupResolver = "group-resolver" DiscoveryGraphNodeTypeResolver = "resolver"
) )
// DiscoveryGraphNode is a single node of the compiled discovery chain. // DiscoveryGraphNode is a single node in the compiled discovery chain.
type DiscoveryGraphNode struct { type DiscoveryGraphNode struct {
Type string Type string
Name string // default chain/service name at this spot Name string // this is NOT necessarily a service
// fields for Type==router // fields for Type==router
Routes []*DiscoveryRoute `json:",omitempty"` Routes []*DiscoveryRoute `json:",omitempty"`
@ -98,34 +82,48 @@ type DiscoveryGraphNode struct {
// fields for Type==splitter // fields for Type==splitter
Splits []*DiscoverySplit `json:",omitempty"` Splits []*DiscoverySplit `json:",omitempty"`
// fields for Type==group-resolver // fields for Type==resolver
GroupResolver *DiscoveryGroupResolver `json:",omitempty"` Resolver *DiscoveryResolver `json:",omitempty"`
} }
// compiled form of ServiceResolverConfigEntry but customized per non-failover target func (s *DiscoveryGraphNode) ServiceName() string {
type DiscoveryGroupResolver struct { if s.Type == DiscoveryGraphNodeTypeResolver {
return s.Resolver.Target.Service
}
return s.Name
}
func (s *DiscoveryGraphNode) MapKey() string {
return fmt.Sprintf("%s:%s", s.Type, s.Name)
}
// compiled form of ServiceResolverConfigEntry
type DiscoveryResolver struct {
Definition *ServiceResolverConfigEntry `json:",omitempty"` Definition *ServiceResolverConfigEntry `json:",omitempty"`
Default bool `json:",omitempty"` Default bool `json:",omitempty"`
ConnectTimeout time.Duration `json:",omitempty"` ConnectTimeout time.Duration `json:",omitempty"`
MeshGateway MeshGatewayConfig `json:",omitempty"`
Target DiscoveryTarget `json:",omitempty"` Target DiscoveryTarget `json:",omitempty"`
Failover *DiscoveryFailover `json:",omitempty"` Failover *DiscoveryFailover `json:",omitempty"`
} }
type DiscoveryTargetConfig struct {
MeshGateway MeshGatewayConfig `json:",omitempty"`
Subset ServiceResolverSubset `json:",omitempty"`
}
// compiled form of ServiceRoute // compiled form of ServiceRoute
type DiscoveryRoute struct { type DiscoveryRoute struct {
Definition *ServiceRoute `json:",omitempty"` Definition *ServiceRoute `json:",omitempty"`
DestinationNode *DiscoveryGraphNode `json:",omitempty"` NextNode string `json:",omitempty"`
} }
// compiled form of ServiceSplit // compiled form of ServiceSplit
type DiscoverySplit struct { type DiscoverySplit struct {
Weight float32 `json:",omitempty"` Weight float32 `json:",omitempty"`
Node *DiscoveryGraphNode `json:",omitempty"` NextNode string `json:",omitempty"`
} }
// compiled form of ServiceResolverFailover // compiled form of ServiceResolverFailover
// TODO(rb): figure out how to get mesh gateways in here
type DiscoveryFailover struct { type DiscoveryFailover struct {
Definition *ServiceResolverFailover `json:",omitempty"` Definition *ServiceResolverFailover `json:",omitempty"`
Targets []DiscoveryTarget `json:",omitempty"` Targets []DiscoveryTarget `json:",omitempty"`
@ -183,17 +181,21 @@ var _ encoding.TextUnmarshaler = (*DiscoveryTarget)(nil)
// //
// This should NOT return any errors. // This should NOT return any errors.
func (t DiscoveryTarget) MarshalText() (text []byte, err error) { func (t DiscoveryTarget) MarshalText() (text []byte, err error) {
return []byte(t.Identifier()), nil
}
func (t DiscoveryTarget) Identifier() string {
var buf bytes.Buffer var buf bytes.Buffer
buf.WriteString(url.QueryEscape(t.Service)) buf.WriteString(url.QueryEscape(t.Service))
buf.WriteRune(',') buf.WriteRune(',')
buf.WriteString(url.QueryEscape(t.ServiceSubset)) buf.WriteString(url.QueryEscape(t.ServiceSubset)) // TODO(rb): move this first so the scoping flows from small->large?
buf.WriteRune(',') buf.WriteRune(',')
if t.Namespace != "default" { if t.Namespace != "default" {
buf.WriteString(url.QueryEscape(t.Namespace)) buf.WriteString(url.QueryEscape(t.Namespace))
} }
buf.WriteRune(',') buf.WriteRune(',')
buf.WriteString(url.QueryEscape(t.Datacenter)) buf.WriteString(url.QueryEscape(t.Datacenter))
return buf.Bytes(), nil return buf.String()
} }
// UnmarshalText implements encoding.TextUnmarshaler. // UnmarshalText implements encoding.TextUnmarshaler.
@ -256,29 +258,3 @@ func (t DiscoveryTarget) String() string {
return b.String() return b.String()
} }
type DiscoveryTargets []DiscoveryTarget
func (targets DiscoveryTargets) Sort() {
sort.Slice(targets, func(i, j int) bool {
if targets[i].Service < targets[j].Service {
return true
} else if targets[i].Service > targets[j].Service {
return false
}
if targets[i].ServiceSubset < targets[j].ServiceSubset {
return true
} else if targets[i].ServiceSubset > targets[j].ServiceSubset {
return false
}
if targets[i].Namespace < targets[j].Namespace {
return true
} else if targets[i].Namespace > targets[j].Namespace {
return false
}
return targets[i].Datacenter < targets[j].Datacenter
})
}

View File

@ -234,7 +234,6 @@ func (s *Server) makeUpstreamClustersForDiscoveryChain(
chain *structs.CompiledDiscoveryChain, chain *structs.CompiledDiscoveryChain,
cfgSnap *proxycfg.ConfigSnapshot, cfgSnap *proxycfg.ConfigSnapshot,
) ([]*envoy.Cluster, error) { ) ([]*envoy.Cluster, error) {
cfg, err := ParseUpstreamConfigNoDefaults(upstream.Config) cfg, err := ParseUpstreamConfigNoDefaults(upstream.Config)
if err != nil { if err != nil {
// Don't hard fail on a config typo, just warn. The parse func returns // Don't hard fail on a config typo, just warn. The parse func returns
@ -250,8 +249,12 @@ func (s *Server) makeUpstreamClustersForDiscoveryChain(
// TODO(rb): make escape hatches work with chains // TODO(rb): make escape hatches work with chains
var out []*envoy.Cluster var out []*envoy.Cluster
for target, node := range chain.GroupResolverNodes {
groupResolver := node.GroupResolver for _, node := range chain.Nodes {
if node.Type != structs.DiscoveryGraphNodeTypeResolver {
continue
}
target := node.Resolver.Target
sni := TargetSNI(target, cfgSnap) sni := TargetSNI(target, cfgSnap)
clusterName := CustomizeClusterName(sni, chain) clusterName := CustomizeClusterName(sni, chain)
@ -260,14 +263,13 @@ func (s *Server) makeUpstreamClustersForDiscoveryChain(
c := &envoy.Cluster{ c := &envoy.Cluster{
Name: clusterName, Name: clusterName,
AltStatName: clusterName, AltStatName: clusterName,
ConnectTimeout: groupResolver.ConnectTimeout, ConnectTimeout: node.Resolver.ConnectTimeout,
ClusterDiscoveryType: &envoy.Cluster_Type{Type: envoy.Cluster_EDS}, ClusterDiscoveryType: &envoy.Cluster_Type{Type: envoy.Cluster_EDS},
CommonLbConfig: &envoy.Cluster_CommonLbConfig{ CommonLbConfig: &envoy.Cluster_CommonLbConfig{
HealthyPanicThreshold: &envoytype.Percent{ HealthyPanicThreshold: &envoytype.Percent{
Value: 0, // disable panic threshold Value: 0, // disable panic threshold
}, },
}, },
// TODO(rb): adjust load assignment
EdsClusterConfig: &envoy.Cluster_EdsClusterConfig{ EdsClusterConfig: &envoy.Cluster_EdsClusterConfig{
EdsConfig: &envoycore.ConfigSource{ EdsConfig: &envoycore.ConfigSource{
ConfigSourceSpecifier: &envoycore.ConfigSource_Ads{ ConfigSourceSpecifier: &envoycore.ConfigSource_Ads{

View File

@ -73,15 +73,21 @@ func (s *Server) endpointsFromSnapshotConnectProxy(cfgSnap *proxycfg.ConfigSnaps
continue // skip the upstream (should not happen) continue // skip the upstream (should not happen)
} }
for target, node := range chain.GroupResolverNodes { // Find all resolver nodes.
groupResolver := node.GroupResolver for _, node := range chain.Nodes {
failover := groupResolver.Failover if node.Type != structs.DiscoveryGraphNodeTypeResolver {
continue
}
failover := node.Resolver.Failover
target := node.Resolver.Target
endpoints, ok := chainEndpointMap[target] endpoints, ok := chainEndpointMap[target]
if !ok { if !ok {
continue // skip the cluster (should not happen) continue // skip the cluster (should not happen)
} }
targetConfig := chain.Targets[target]
var ( var (
endpointGroups []loadAssignmentEndpointGroup endpointGroups []loadAssignmentEndpointGroup
overprovisioningFactor int overprovisioningFactor int
@ -89,7 +95,7 @@ func (s *Server) endpointsFromSnapshotConnectProxy(cfgSnap *proxycfg.ConfigSnaps
primaryGroup := loadAssignmentEndpointGroup{ primaryGroup := loadAssignmentEndpointGroup{
Endpoints: endpoints, Endpoints: endpoints,
OnlyPassing: chain.SubsetDefinitionForTarget(target).OnlyPassing, OnlyPassing: targetConfig.Subset.OnlyPassing,
} }
if failover != nil && len(failover.Targets) > 0 { if failover != nil && len(failover.Targets) > 0 {
@ -112,9 +118,11 @@ func (s *Server) endpointsFromSnapshotConnectProxy(cfgSnap *proxycfg.ConfigSnaps
continue // skip the failover target (should not happen) continue // skip the failover target (should not happen)
} }
failTargetConfig := chain.Targets[failTarget]
endpointGroups = append(endpointGroups, loadAssignmentEndpointGroup{ endpointGroups = append(endpointGroups, loadAssignmentEndpointGroup{
Endpoints: failEndpoints, Endpoints: failEndpoints,
OnlyPassing: chain.SubsetDefinitionForTarget(failTarget).OnlyPassing, OnlyPassing: failTargetConfig.Subset.OnlyPassing,
}) })
} }
} else { } else {

View File

@ -266,11 +266,11 @@ func Test_endpointsFromSnapshot(t *testing.T) {
Namespace: "default", Namespace: "default",
Datacenter: "dc1", Datacenter: "dc1",
} }
dbResolverNode := chain.GroupResolverNodes[dbTarget] dbResolverNode := chain.Nodes["resolver:"+dbTarget.Identifier()]
groupResolverFailover := dbResolverNode.GroupResolver.Failover failover := dbResolverNode.Resolver.Failover
groupResolverFailover.Definition.OverprovisioningFactor = 160 failover.Definition.OverprovisioningFactor = 160
}, },
}, },
{ {

View File

@ -73,11 +73,16 @@ func makeUpstreamRouteForDiscoveryChain(
var routes []envoyroute.Route var routes []envoyroute.Route
switch chain.Node.Type { startNode := chain.Nodes[chain.StartNode]
case structs.DiscoveryGraphNodeTypeRouter: if startNode == nil {
routes = make([]envoyroute.Route, 0, len(chain.Node.Routes)) panic("missing first node in compiled discovery chain for: " + chain.ServiceName)
}
for _, discoveryRoute := range chain.Node.Routes { switch startNode.Type {
case structs.DiscoveryGraphNodeTypeRouter:
routes = make([]envoyroute.Route, 0, len(startNode.Routes))
for _, discoveryRoute := range startNode.Routes {
routeMatch := makeRouteMatchForDiscoveryRoute(discoveryRoute, chain.Protocol) routeMatch := makeRouteMatchForDiscoveryRoute(discoveryRoute, chain.Protocol)
var ( var (
@ -85,19 +90,19 @@ func makeUpstreamRouteForDiscoveryChain(
err error err error
) )
next := discoveryRoute.DestinationNode nextNode := chain.Nodes[discoveryRoute.NextNode]
if next.Type == structs.DiscoveryGraphNodeTypeSplitter { switch nextNode.Type {
routeAction, err = makeRouteActionForSplitter(next.Splits, chain, cfgSnap) case structs.DiscoveryGraphNodeTypeSplitter:
routeAction, err = makeRouteActionForSplitter(nextNode.Splits, chain, cfgSnap)
if err != nil { if err != nil {
return nil, err return nil, err
} }
} else if next.Type == structs.DiscoveryGraphNodeTypeGroupResolver { case structs.DiscoveryGraphNodeTypeResolver:
groupResolver := next.GroupResolver routeAction = makeRouteActionForSingleCluster(nextNode.Resolver.Target, chain, cfgSnap)
routeAction = makeRouteActionForSingleCluster(groupResolver.Target, chain, cfgSnap)
} else { default:
return nil, fmt.Errorf("unexpected graph node after route %q", next.Type) return nil, fmt.Errorf("unexpected graph node after route %q", nextNode.Type)
} }
// TODO(rb): Better help handle the envoy case where you need (prefix=/foo/,rewrite=/) and (exact=/foo,rewrite=/) to do a full rewrite // TODO(rb): Better help handle the envoy case where you need (prefix=/foo/,rewrite=/) and (exact=/foo,rewrite=/) to do a full rewrite
@ -142,7 +147,7 @@ func makeUpstreamRouteForDiscoveryChain(
} }
case structs.DiscoveryGraphNodeTypeSplitter: case structs.DiscoveryGraphNodeTypeSplitter:
routeAction, err := makeRouteActionForSplitter(chain.Node.Splits, chain, cfgSnap) routeAction, err := makeRouteActionForSplitter(startNode.Splits, chain, cfgSnap)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -154,10 +159,8 @@ func makeUpstreamRouteForDiscoveryChain(
routes = []envoyroute.Route{defaultRoute} routes = []envoyroute.Route{defaultRoute}
case structs.DiscoveryGraphNodeTypeGroupResolver: case structs.DiscoveryGraphNodeTypeResolver:
groupResolver := chain.Node.GroupResolver routeAction := makeRouteActionForSingleCluster(startNode.Resolver.Target, chain, cfgSnap)
routeAction := makeRouteActionForSingleCluster(groupResolver.Target, chain, cfgSnap)
defaultRoute := envoyroute.Route{ defaultRoute := envoyroute.Route{
Match: makeDefaultRouteMatch(), Match: makeDefaultRouteMatch(),
@ -167,7 +170,7 @@ func makeUpstreamRouteForDiscoveryChain(
routes = []envoyroute.Route{defaultRoute} routes = []envoyroute.Route{defaultRoute}
default: default:
panic("unknown top node in discovery chain of type: " + chain.Node.Type) panic("unknown first node in discovery chain of type: " + startNode.Type)
} }
return &envoy.RouteConfiguration{ return &envoy.RouteConfiguration{
@ -320,11 +323,12 @@ func makeRouteActionForSingleCluster(target structs.DiscoveryTarget, chain *stru
func makeRouteActionForSplitter(splits []*structs.DiscoverySplit, chain *structs.CompiledDiscoveryChain, cfgSnap *proxycfg.ConfigSnapshot) (*envoyroute.Route_Route, error) { func makeRouteActionForSplitter(splits []*structs.DiscoverySplit, chain *structs.CompiledDiscoveryChain, cfgSnap *proxycfg.ConfigSnapshot) (*envoyroute.Route_Route, error) {
clusters := make([]*envoyroute.WeightedCluster_ClusterWeight, 0, len(splits)) clusters := make([]*envoyroute.WeightedCluster_ClusterWeight, 0, len(splits))
for _, split := range splits { for _, split := range splits {
if split.Node.Type != structs.DiscoveryGraphNodeTypeGroupResolver { nextNode := chain.Nodes[split.NextNode]
return nil, fmt.Errorf("unexpected splitter destination node type: %s", split.Node.Type)
if nextNode.Type != structs.DiscoveryGraphNodeTypeResolver {
return nil, fmt.Errorf("unexpected splitter destination node type: %s", nextNode.Type)
} }
groupResolver := split.Node.GroupResolver target := nextNode.Resolver.Target
target := groupResolver.Target
sni := TargetSNI(target, cfgSnap) sni := TargetSNI(target, cfgSnap)
clusterName := CustomizeClusterName(sni, chain) clusterName := CustomizeClusterName(sni, chain)