Skip to content

Commit 9a8bc10

Browse files
authored
xds: unify client and server handling HttpConnectionManager (#8228)
Enables parsing HttpConnectionManager filter for the server side TCP listener, with the same codepath for handling it on the client side. Major changes include: - Remodeled LdsUpdate with HttpConnectionManager. Now LdsUpdate is an oneof of HttpConnectionManager (for client side) or Listener (for server side). Each of Listener's FiliterChain contains an HttpConnectionManager (required). Refactored code for validating and parsing the TCP Listener (for server side), put it into ClientXdsClient. The common part of validating/parsing HttpConnectionManager is reused/shared for client side. - Included the name of FilterChain in the parsed form. As specified by the API, each FilterChain has a unique name. If the name is not provided by the control plane, a UUID is used. FilterChain names can be used for bookkeeping a set of FilterChain easily (e.g., used as map key). - Added methods isSupportedOnClients() and isSupportedOnServers() to the Filter interface. Parsing the top-level HttpFilter requires knowing if the HttpFilter implementation is supported for the target usage (client-side or server-side). Note, parsing override HttpFilter configs does not need to know whether the config is used for an HttpFilter that is only supported for the client-side or server side. - Added a new kind of Route: Route with non-forwarding action. Updated the XdsNameResolver being able to handle Route with non-forwarding action: if such a Route is matched to an RPC, that RPC is failed. Note, it is possible that XdsNameResolver receives xDS updates with all Routes with non-forwarding action. That is, the service config will not reference any cluster. Such case can be handled by cluster_manager LB policy's LB config parser: the parser returns the error to Channel and the Channel will handle it as error service config.
1 parent 2258d2e commit 9a8bc10

20 files changed

+1485
-1347
lines changed

xds/src/main/java/io/grpc/xds/ClientXdsClient.java

+340-141
Large diffs are not rendered by default.

xds/src/main/java/io/grpc/xds/EnvoyServerProtoData.java

+47-235
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
/*
2+
* Copyright 2021 The gRPC Authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package io.grpc.xds;
18+
19+
import static com.google.common.base.Preconditions.checkNotNull;
20+
21+
import com.google.auto.value.AutoValue;
22+
import com.google.common.collect.ImmutableList;
23+
import io.grpc.xds.Filter.NamedFilterConfig;
24+
import java.util.List;
25+
import javax.annotation.Nullable;
26+
27+
/**
28+
* HttpConnectionManager is a network filter for proxying HTTP requests.
29+
*/
30+
@AutoValue
31+
abstract class HttpConnectionManager {
32+
// Total number of nanoseconds to keep alive an HTTP request/response stream.
33+
abstract long httpMaxStreamDurationNano();
34+
35+
// Name of the route configuration to be used for RDS resource discovery.
36+
@Nullable
37+
abstract String rdsName();
38+
39+
// List of virtual hosts that make up the route table.
40+
@Nullable
41+
abstract ImmutableList<VirtualHost> virtualHosts();
42+
43+
// List of http filter configs. Null if HttpFilter support is not enabled.
44+
@Nullable
45+
abstract ImmutableList<NamedFilterConfig> httpFilterConfigs();
46+
47+
static HttpConnectionManager forRdsName(long httpMaxStreamDurationNano, String rdsName,
48+
@Nullable List<NamedFilterConfig> httpFilterConfigs) {
49+
checkNotNull(rdsName, "rdsName");
50+
return create(httpMaxStreamDurationNano, rdsName, null, httpFilterConfigs);
51+
}
52+
53+
static HttpConnectionManager forVirtualHosts(long httpMaxStreamDurationNano,
54+
List<VirtualHost> virtualHosts, @Nullable List<NamedFilterConfig> httpFilterConfigs) {
55+
checkNotNull(virtualHosts, "virtualHosts");
56+
return create(httpMaxStreamDurationNano, null, virtualHosts,
57+
httpFilterConfigs);
58+
}
59+
60+
private static HttpConnectionManager create(long httpMaxStreamDurationNano,
61+
@Nullable String rdsName, @Nullable List<VirtualHost> virtualHosts,
62+
@Nullable List<NamedFilterConfig> httpFilterConfigs) {
63+
return new AutoValue_HttpConnectionManager(
64+
httpMaxStreamDurationNano, rdsName,
65+
virtualHosts == null ? null : ImmutableList.copyOf(virtualHosts),
66+
httpFilterConfigs == null ? null : ImmutableList.copyOf(httpFilterConfigs));
67+
}
68+
}

xds/src/main/java/io/grpc/xds/RouterFilter.java

+10-1
Original file line numberDiff line numberDiff line change
@@ -19,14 +19,16 @@
1919
import com.google.protobuf.Message;
2020
import io.grpc.ClientInterceptor;
2121
import io.grpc.LoadBalancer.PickSubchannelArgs;
22+
import io.grpc.ServerInterceptor;
2223
import io.grpc.xds.Filter.ClientInterceptorBuilder;
24+
import io.grpc.xds.Filter.ServerInterceptorBuilder;
2325
import java.util.concurrent.ScheduledExecutorService;
2426
import javax.annotation.Nullable;
2527

2628
/**
2729
* Router filter implementation. Currently this filter does not parse any field in the config.
2830
*/
29-
enum RouterFilter implements Filter, ClientInterceptorBuilder {
31+
enum RouterFilter implements Filter, ClientInterceptorBuilder, ServerInterceptorBuilder {
3032
INSTANCE;
3133

3234
static final String TYPE_URL =
@@ -66,4 +68,11 @@ public ClientInterceptor buildClientInterceptor(
6668
ScheduledExecutorService scheduler) {
6769
return null;
6870
}
71+
72+
@Nullable
73+
@Override
74+
public ServerInterceptor buildServerInterceptor(
75+
FilterConfig config, @Nullable Filter.FilterConfig overrideConfig) {
76+
return null;
77+
}
6978
}

xds/src/main/java/io/grpc/xds/VirtualHost.java

+13-2
Original file line numberDiff line numberDiff line change
@@ -57,12 +57,23 @@ public static VirtualHost create(
5757
abstract static class Route {
5858
abstract RouteMatch routeMatch();
5959

60+
@Nullable
6061
abstract RouteAction routeAction();
6162

6263
abstract ImmutableMap<String, FilterConfig> filterConfigOverrides();
6364

64-
static Route create(
65-
RouteMatch routeMatch, RouteAction routeAction,
65+
static Route forAction(RouteMatch routeMatch, RouteAction routeAction,
66+
Map<String, FilterConfig> filterConfigOverrides) {
67+
return create(routeMatch, routeAction, filterConfigOverrides);
68+
}
69+
70+
static Route forNonForwardingAction(RouteMatch routeMatch,
71+
Map<String, FilterConfig> filterConfigOverrides) {
72+
return create(routeMatch, null, filterConfigOverrides);
73+
}
74+
75+
private static Route create(
76+
RouteMatch routeMatch, @Nullable RouteAction routeAction,
6677
Map<String, FilterConfig> filterConfigOverrides) {
6778
return new AutoValue_VirtualHost_Route(
6879
routeMatch, routeAction, ImmutableMap.copyOf(filterConfigOverrides));

xds/src/main/java/io/grpc/xds/XdsClient.java

+13-82
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,6 @@
2020

2121
import com.google.auto.value.AutoValue;
2222
import com.google.common.base.MoreObjects;
23-
import com.google.common.base.MoreObjects.ToStringHelper;
2423
import com.google.common.collect.ImmutableList;
2524
import com.google.protobuf.Any;
2625
import io.grpc.Status;
@@ -29,7 +28,6 @@
2928
import io.grpc.xds.Endpoints.LocalityLbEndpoints;
3029
import io.grpc.xds.EnvoyServerProtoData.Listener;
3130
import io.grpc.xds.EnvoyServerProtoData.UpstreamTlsContext;
32-
import io.grpc.xds.Filter.NamedFilterConfig;
3331
import io.grpc.xds.LoadStatsManager2.ClusterDropStats;
3432
import io.grpc.xds.LoadStatsManager2.ClusterLocalityStats;
3533
import java.util.ArrayList;
@@ -48,92 +46,25 @@
4846
*/
4947
abstract class XdsClient {
5048

51-
static final class LdsUpdate implements ResourceUpdate {
52-
// Total number of nanoseconds to keep alive an HTTP request/response stream.
53-
final long httpMaxStreamDurationNano;
54-
// The name of the route configuration to be used for RDS resource discovery.
55-
@Nullable
56-
final String rdsName;
57-
// The list virtual hosts that make up the route table.
58-
@Nullable
59-
final List<VirtualHost> virtualHosts;
60-
// Filter instance names. Null if HttpFilter support is not enabled.
61-
@Nullable final List<NamedFilterConfig> filterChain;
62-
// Server side Listener.
49+
@AutoValue
50+
abstract static class LdsUpdate implements ResourceUpdate {
51+
// Http level api listener configuration.
6352
@Nullable
64-
final Listener listener;
65-
66-
LdsUpdate(
67-
long httpMaxStreamDurationNano, String rdsName,
68-
@Nullable List<NamedFilterConfig> filterChain) {
69-
this(httpMaxStreamDurationNano, rdsName, null, filterChain);
70-
}
71-
72-
LdsUpdate(
73-
long httpMaxStreamDurationNano, List<VirtualHost> virtualHosts,
74-
@Nullable List<NamedFilterConfig> filterChain) {
75-
this(httpMaxStreamDurationNano, null, virtualHosts, filterChain);
76-
}
77-
78-
private LdsUpdate(
79-
long httpMaxStreamDurationNano, @Nullable String rdsName,
80-
@Nullable List<VirtualHost> virtualHosts, @Nullable List<NamedFilterConfig> filterChain) {
81-
this.httpMaxStreamDurationNano = httpMaxStreamDurationNano;
82-
this.rdsName = rdsName;
83-
this.virtualHosts = virtualHosts == null
84-
? null : Collections.unmodifiableList(new ArrayList<>(virtualHosts));
85-
this.filterChain = filterChain == null ? null : Collections.unmodifiableList(filterChain);
86-
this.listener = null;
87-
}
88-
89-
LdsUpdate(Listener listener) {
90-
this.listener = listener;
91-
this.httpMaxStreamDurationNano = 0L;
92-
this.rdsName = null;
93-
this.filterChain = null;
94-
this.virtualHosts = null;
95-
}
53+
abstract HttpConnectionManager httpConnectionManager();
9654

97-
@Override
98-
public int hashCode() {
99-
return Objects.hash(
100-
httpMaxStreamDurationNano, rdsName, virtualHosts, filterChain, listener);
101-
}
55+
// Tcp level listener configuration.
56+
@Nullable
57+
abstract Listener listener();
10258

103-
@Override
104-
public boolean equals(Object o) {
105-
if (this == o) {
106-
return true;
107-
}
108-
if (o == null || getClass() != o.getClass()) {
109-
return false;
110-
}
111-
LdsUpdate that = (LdsUpdate) o;
112-
return httpMaxStreamDurationNano == that.httpMaxStreamDurationNano
113-
&& Objects.equals(rdsName, that.rdsName)
114-
&& Objects.equals(virtualHosts, that.virtualHosts)
115-
&& Objects.equals(filterChain, that.filterChain)
116-
&& Objects.equals(listener, that.listener);
59+
static LdsUpdate forApiListener(HttpConnectionManager httpConnectionManager) {
60+
checkNotNull(httpConnectionManager, "httpConnectionManager");
61+
return new AutoValue_XdsClient_LdsUpdate(httpConnectionManager, null);
11762
}
11863

119-
@Override
120-
public String toString() {
121-
ToStringHelper toStringHelper = MoreObjects.toStringHelper(this);
122-
toStringHelper.add("httpMaxStreamDurationNano", httpMaxStreamDurationNano);
123-
if (rdsName != null) {
124-
toStringHelper.add("rdsName", rdsName);
125-
} else {
126-
toStringHelper.add("virtualHosts", virtualHosts);
127-
}
128-
if (filterChain != null) {
129-
toStringHelper.add("filterChain", filterChain);
130-
}
131-
if (listener != null) {
132-
toStringHelper.add("listener", listener);
133-
}
134-
return toStringHelper.toString();
64+
static LdsUpdate forTcpListener(Listener listener) {
65+
checkNotNull(listener, "listener");
66+
return new AutoValue_XdsClient_LdsUpdate(null, listener);
13567
}
136-
13768
}
13869

13970
static final class RdsUpdate implements ResourceUpdate {

xds/src/main/java/io/grpc/xds/XdsClientWrapperForServerSds.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ public void start() {
114114
new XdsClient.LdsResourceWatcher() {
115115
@Override
116116
public void onChanged(XdsClient.LdsUpdate update) {
117-
releaseOldSuppliers(curListener.getAndSet(update.listener));
117+
releaseOldSuppliers(curListener.getAndSet(update.listener()));
118118
reportSuccess();
119119
}
120120

xds/src/main/java/io/grpc/xds/XdsNameResolver.java

+18-9
Original file line numberDiff line numberDiff line change
@@ -359,6 +359,10 @@ public Result selectConfig(PickSubchannelArgs args) {
359359
return Result.forError(
360360
Status.UNAVAILABLE.withDescription("Could not find xDS route matching RPC"));
361361
}
362+
if (selectedRoute.routeAction() == null) {
363+
return Result.forError(Status.UNAVAILABLE.withDescription(
364+
"Could not route RPC to Route with non-forwarding action"));
365+
}
362366
RouteAction action = selectedRoute.routeAction();
363367
if (action.cluster() != null) {
364368
cluster = action.cluster();
@@ -648,14 +652,17 @@ public void run() {
648652
return;
649653
}
650654
logger.log(XdsLogLevel.INFO, "Receive LDS resource update: {0}", update);
651-
List<VirtualHost> virtualHosts = update.virtualHosts;
652-
String rdsName = update.rdsName;
655+
HttpConnectionManager httpConnectionManager = update.httpConnectionManager();
656+
List<VirtualHost> virtualHosts = httpConnectionManager.virtualHosts();
657+
String rdsName = httpConnectionManager.rdsName();
653658
cleanUpRouteDiscoveryState();
654659
if (virtualHosts != null) {
655-
updateRoutes(virtualHosts, update.httpMaxStreamDurationNano, update.filterChain);
660+
updateRoutes(virtualHosts, httpConnectionManager.httpMaxStreamDurationNano(),
661+
httpConnectionManager.httpFilterConfigs());
656662
} else {
657663
routeDiscoveryState = new RouteDiscoveryState(
658-
rdsName, update.httpMaxStreamDurationNano, update.filterChain);
664+
rdsName, httpConnectionManager.httpMaxStreamDurationNano(),
665+
httpConnectionManager.httpFilterConfigs());
659666
logger.log(XdsLogLevel.INFO, "Start watching RDS resource {0}", rdsName);
660667
xdsClient.watchRdsResource(rdsName, routeDiscoveryState);
661668
}
@@ -739,11 +746,13 @@ private void updateRoutes(List<VirtualHost> virtualHosts, long httpMaxStreamDura
739746
Set<String> clusters = new HashSet<>();
740747
for (Route route : routes) {
741748
RouteAction action = route.routeAction();
742-
if (action.cluster() != null) {
743-
clusters.add(action.cluster());
744-
} else if (action.weightedClusters() != null) {
745-
for (ClusterWeight weighedCluster : action.weightedClusters()) {
746-
clusters.add(weighedCluster.name());
749+
if (action != null) {
750+
if (action.cluster() != null) {
751+
clusters.add(action.cluster());
752+
} else if (action.weightedClusters() != null) {
753+
for (ClusterWeight weighedCluster : action.weightedClusters()) {
754+
clusters.add(weighedCluster.name());
755+
}
747756
}
748757
}
749758
}

0 commit comments

Comments
 (0)