• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

grpc / grpc-java / #20320

17 Jun 2026 05:11PM UTC coverage: 88.886% (+0.01%) from 88.874%
#20320

push

github

web-flow
api: Add Grpc.newChannelBuilder accepting NameResolverRegistry (#11901)

This introduces a new `Grpc.newChannelBuilder` overload that allows callers to pass an explicit `NameResolverRegistry`. Fixes #11055

36550 of 41120 relevant lines covered (88.89%)

0.89 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

91.73
/../core/src/main/java/io/grpc/internal/ManagedChannelImplBuilder.java
1
/*
2
 * Copyright 2020 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.internal;
18

19
import static com.google.common.base.Preconditions.checkArgument;
20
import static com.google.common.base.Preconditions.checkNotNull;
21
import static io.grpc.internal.UriWrapper.wrap;
22

23
import com.google.common.annotations.VisibleForTesting;
24
import com.google.common.base.Preconditions;
25
import com.google.common.util.concurrent.MoreExecutors;
26
import com.google.errorprone.annotations.DoNotCall;
27
import io.grpc.Attributes;
28
import io.grpc.BinaryLog;
29
import io.grpc.CallCredentials;
30
import io.grpc.CallOptions;
31
import io.grpc.Channel;
32
import io.grpc.ChannelCredentials;
33
import io.grpc.ClientCall;
34
import io.grpc.ClientInterceptor;
35
import io.grpc.ClientTransportFilter;
36
import io.grpc.CompressorRegistry;
37
import io.grpc.DecompressorRegistry;
38
import io.grpc.EquivalentAddressGroup;
39
import io.grpc.InternalChannelz;
40
import io.grpc.InternalConfiguratorRegistry;
41
import io.grpc.InternalFeatureFlags;
42
import io.grpc.ManagedChannel;
43
import io.grpc.ManagedChannelBuilder;
44
import io.grpc.MethodDescriptor;
45
import io.grpc.MetricSink;
46
import io.grpc.NameResolver;
47
import io.grpc.NameResolverProvider;
48
import io.grpc.NameResolverRegistry;
49
import io.grpc.ProxyDetector;
50
import io.grpc.StatusOr;
51
import io.grpc.Uri;
52
import java.lang.reflect.InvocationTargetException;
53
import java.lang.reflect.Method;
54
import java.net.SocketAddress;
55
import java.net.URI;
56
import java.net.URISyntaxException;
57
import java.util.ArrayList;
58
import java.util.Arrays;
59
import java.util.Collection;
60
import java.util.Collections;
61
import java.util.IdentityHashMap;
62
import java.util.LinkedHashMap;
63
import java.util.List;
64
import java.util.Map;
65
import java.util.concurrent.Executor;
66
import java.util.concurrent.TimeUnit;
67
import java.util.logging.Level;
68
import java.util.logging.Logger;
69
import java.util.regex.Pattern;
70
import javax.annotation.Nullable;
71

72
/**
73
 * Default managed channel builder, for usage in Transport implementations.
74
 */
75
public final class ManagedChannelImplBuilder
76
    extends ManagedChannelBuilder<ManagedChannelImplBuilder> {
77
  private static final String DIRECT_ADDRESS_SCHEME = "directaddress";
78

79
  private static final Logger log = Logger.getLogger(ManagedChannelImplBuilder.class.getName());
1✔
80

81
  @DoNotCall("ClientTransportFactoryBuilder is required, use a constructor")
82
  public static ManagedChannelBuilder<?> forAddress(String name, int port) {
83
    throw new UnsupportedOperationException(
×
84
        "ClientTransportFactoryBuilder is required, use a constructor");
85
  }
86

87
  @DoNotCall("ClientTransportFactoryBuilder is required, use a constructor")
88
  public static ManagedChannelBuilder<?> forTarget(String target) {
89
    throw new UnsupportedOperationException(
×
90
        "ClientTransportFactoryBuilder is required, use a constructor");
91
  }
92

93
  /**
94
   * An idle timeout larger than this would disable idle mode.
95
   */
96
  @VisibleForTesting
97
  static final long IDLE_MODE_MAX_TIMEOUT_DAYS = 30;
98

99
  /**
100
   * The default idle timeout.
101
   */
102
  @VisibleForTesting
103
  static final long IDLE_MODE_DEFAULT_TIMEOUT_MILLIS = TimeUnit.MINUTES.toMillis(30);
1✔
104

105
  /**
106
   * An idle timeout smaller than this would be capped to it.
107
   */
108
  static final long IDLE_MODE_MIN_TIMEOUT_MILLIS = TimeUnit.SECONDS.toMillis(1);
1✔
109

110
  private static final ObjectPool<? extends Executor> DEFAULT_EXECUTOR_POOL =
1✔
111
      SharedResourcePool.forResource(GrpcUtil.SHARED_CHANNEL_EXECUTOR);
1✔
112

113
  private static final DecompressorRegistry DEFAULT_DECOMPRESSOR_REGISTRY =
114
      DecompressorRegistry.getDefaultInstance();
1✔
115

116
  private static final CompressorRegistry DEFAULT_COMPRESSOR_REGISTRY =
117
      CompressorRegistry.getDefaultInstance();
1✔
118

119
  private static final long DEFAULT_RETRY_BUFFER_SIZE_IN_BYTES = 1L << 24;  // 16M
120
  private static final long DEFAULT_PER_RPC_BUFFER_LIMIT_IN_BYTES = 1L << 20; // 1M
121

122
  // Matching this pattern means the target string is a URI target or at least intended to be one.
123
  // A URI target must be an absolute hierarchical URI.
124
  // From RFC 2396: scheme = alpha *( alpha | digit | "+" | "-" | "." )
125
  @VisibleForTesting
126
  static final Pattern URI_PATTERN = Pattern.compile("[a-zA-Z][a-zA-Z0-9+.-]*:/.*");
1✔
127

128
  private static final Method GET_CLIENT_INTERCEPTOR_METHOD;
129

130
  static {
131
    Method getClientInterceptorMethod = null;
1✔
132
    try {
133
      Class<?> censusStatsAccessor =
1✔
134
          Class.forName("io.grpc.census.InternalCensusStatsAccessor");
1✔
135
      getClientInterceptorMethod =
1✔
136
          censusStatsAccessor.getDeclaredMethod(
1✔
137
              "getClientInterceptor",
138
              boolean.class,
139
              boolean.class,
140
              boolean.class,
141
              boolean.class);
142
    } catch (ClassNotFoundException e) {
1✔
143
      // Replace these separate catch statements with multicatch when Android min-API >= 19
144
      log.log(Level.FINE, "Unable to apply census stats", e);
1✔
145
    } catch (NoSuchMethodException e) {
×
146
      log.log(Level.FINE, "Unable to apply census stats", e);
×
147
    }
1✔
148
    GET_CLIENT_INTERCEPTOR_METHOD = getClientInterceptorMethod;
1✔
149
  }
1✔
150

151

152
  ObjectPool<? extends Executor> executorPool = DEFAULT_EXECUTOR_POOL;
1✔
153

154
  ObjectPool<? extends Executor> offloadExecutorPool = DEFAULT_EXECUTOR_POOL;
1✔
155

156
  private final List<ClientInterceptor> interceptors = new ArrayList<>();
1✔
157
  NameResolverRegistry nameResolverRegistry = NameResolverRegistry.getDefaultRegistry();
1✔
158

159
  @Nullable
160
  NameResolverProvider nameResolverProvider;
161

162
  final List<ClientTransportFilter> transportFilters = new ArrayList<>();
1✔
163

164
  final String target;
165
  @Nullable
166
  final ChannelCredentials channelCredentials;
167
  @Nullable
168
  final CallCredentials callCredentials;
169
  @Nullable
170
  IdentityHashMap<NameResolver.Args.Key<?>, Object> nameResolverCustomArgs;
171

172
  @Nullable
173
  private final SocketAddress directServerAddress;
174

175
  @Nullable
176
  String userAgent;
177

178
  @Nullable
179
  String authorityOverride;
180

181
  String defaultLbPolicy = GrpcUtil.DEFAULT_LB_POLICY;
1✔
182

183
  boolean fullStreamDecompression;
184

185
  DecompressorRegistry decompressorRegistry = DEFAULT_DECOMPRESSOR_REGISTRY;
1✔
186

187
  CompressorRegistry compressorRegistry = DEFAULT_COMPRESSOR_REGISTRY;
1✔
188

189
  long idleTimeoutMillis = IDLE_MODE_DEFAULT_TIMEOUT_MILLIS;
1✔
190

191
  int maxRetryAttempts = 5;
1✔
192
  int maxHedgedAttempts = 5;
1✔
193
  long retryBufferSize = DEFAULT_RETRY_BUFFER_SIZE_IN_BYTES;
1✔
194
  long perRpcBufferLimit = DEFAULT_PER_RPC_BUFFER_LIMIT_IN_BYTES;
1✔
195
  boolean retryEnabled = true;
1✔
196

197
  InternalChannelz channelz = InternalChannelz.instance();
1✔
198
  int maxTraceEvents;
199

200
  @Nullable
201
  Map<String, ?> defaultServiceConfig;
202
  boolean lookUpServiceConfig = true;
1✔
203

204
  @Nullable
205
  BinaryLog binlog;
206

207
  @Nullable
208
  ProxyDetector proxyDetector;
209

210
  private boolean authorityCheckerDisabled;
211
  private boolean statsEnabled = true;
1✔
212
  private boolean recordStartedRpcs = true;
1✔
213
  private boolean recordFinishedRpcs = true;
1✔
214
  private boolean recordRealTimeMetrics = false;
1✔
215
  private boolean recordRetryMetrics = true;
1✔
216
  private boolean tracingEnabled = true;
1✔
217
  List<MetricSink> metricSinks = new ArrayList<>();
1✔
218

219
  /**
220
   * An interface for Transport implementors to provide the {@link ClientTransportFactory}
221
   * appropriate for the channel.
222
   */
223
  public interface ClientTransportFactoryBuilder {
224
    ClientTransportFactory buildClientTransportFactory();
225
  }
226

227
  /**
228
   * Convenience ClientTransportFactoryBuilder, throws UnsupportedOperationException().
229
   */
230
  public static class UnsupportedClientTransportFactoryBuilder implements
1✔
231
      ClientTransportFactoryBuilder {
232
    @Override
233
    public ClientTransportFactory buildClientTransportFactory() {
234
      throw new UnsupportedOperationException();
×
235
    }
236
  }
237

238
  /**
239
   * An interface for Transport implementors to provide a default port to {@link
240
   * io.grpc.NameResolver} for use in cases where the target string doesn't include a port. The
241
   * default implementation returns {@link GrpcUtil#DEFAULT_PORT_SSL}.
242
   */
243
  public interface ChannelBuilderDefaultPortProvider {
244
    int getDefaultPort();
245
  }
246

247
  /**
248
   * Default implementation of {@link ChannelBuilderDefaultPortProvider} that returns a fixed port.
249
   */
250
  public static final class FixedPortProvider implements ChannelBuilderDefaultPortProvider {
251
    private final int port;
252

253
    public FixedPortProvider(int port) {
1✔
254
      this.port = port;
1✔
255
    }
1✔
256

257
    @Override
258
    public int getDefaultPort() {
259
      return port;
1✔
260
    }
261
  }
262

263
  private static final class ManagedChannelDefaultPortProvider implements
264
      ChannelBuilderDefaultPortProvider {
265
    @Override
266
    public int getDefaultPort() {
267
      return GrpcUtil.DEFAULT_PORT_SSL;
1✔
268
    }
269
  }
270

271
  private final ClientTransportFactoryBuilder clientTransportFactoryBuilder;
272
  private final ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider;
273

274
  /**
275
   * Creates a new managed channel builder with a target string, which can be either a valid {@link
276
   * io.grpc.NameResolver}-compliant URI, or an authority string. Transport implementors must
277
   * provide client transport factory builder, and may set custom channel default port provider.
278
   */
279
  public ManagedChannelImplBuilder(String target,
280
      ClientTransportFactoryBuilder clientTransportFactoryBuilder,
281
      @Nullable ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider) {
282
    this(target, null, null, clientTransportFactoryBuilder, channelBuilderDefaultPortProvider);
1✔
283
  }
1✔
284

285
  /**
286
   * Creates a new managed channel builder with a target string, which can be either a valid {@link
287
   * io.grpc.NameResolver}-compliant URI, or an authority string. Transport implementors must
288
   * provide client transport factory builder, and may set custom channel default port provider.
289
   *
290
   * @param channelCreds The ChannelCredentials provided by the user. These may be used when
291
   *     creating derivative channels.
292
   */
293
  public ManagedChannelImplBuilder(
294
      String target, @Nullable ChannelCredentials channelCreds, @Nullable CallCredentials callCreds,
295
      ClientTransportFactoryBuilder clientTransportFactoryBuilder,
296
      @Nullable ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider) {
297
    this(
1✔
298
        target,
299
        channelCreds,
300
        callCreds,
301
        clientTransportFactoryBuilder,
302
        channelBuilderDefaultPortProvider,
303
        null,
304
        null);
305
  }
1✔
306

307
  /**
308
   * Creates a new managed channel builder with a target string, which can be
309
   * either a valid {@link io.grpc.NameResolver}-compliant URI, or an authority
310
   * string. Transport
311
   * implementors must provide client transport factory builder, and may set
312
   * custom channel default
313
   * port provider.
314
   *
315
   * @param channelCreds         The ChannelCredentials provided by the user.
316
   *                             These may be used when
317
   *                             creating derivative channels.
318
   * @param nameResolverRegistry the registry used to look up name resolvers.
319
   * @param nameResolverProvider the provider used to look up name resolvers.
320
   */
321
  public ManagedChannelImplBuilder(
322
      String target, @Nullable ChannelCredentials channelCreds, @Nullable CallCredentials callCreds,
323
      ClientTransportFactoryBuilder clientTransportFactoryBuilder,
324
      @Nullable ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider,
325
      @Nullable NameResolverRegistry nameResolverRegistry,
326
      @Nullable NameResolverProvider nameResolverProvider) {
1✔
327
    this.target = checkNotNull(target, "target");
1✔
328
    this.channelCredentials = channelCreds;
1✔
329
    this.callCredentials = callCreds;
1✔
330
    this.clientTransportFactoryBuilder = checkNotNull(clientTransportFactoryBuilder,
1✔
331
        "clientTransportFactoryBuilder");
332
    this.directServerAddress = null;
1✔
333

334
    this.channelBuilderDefaultPortProvider =
1✔
335
        channelBuilderDefaultPortProvider != null
1✔
336
            ? channelBuilderDefaultPortProvider
1✔
337
            : new ManagedChannelDefaultPortProvider();
1✔
338
    this.nameResolverRegistry =
1✔
339
        nameResolverRegistry != null
1✔
340
            ? nameResolverRegistry
1✔
341
            : NameResolverRegistry.getDefaultRegistry();
1✔
342
    this.nameResolverProvider = nameResolverProvider;
1✔
343

344
    // TODO(dnvindhya): Move configurator to all the individual builders
345
    InternalConfiguratorRegistry.configureChannelBuilder(this);
1✔
346
  }
1✔
347

348
  /**
349
   * Returns a target string for the SocketAddress. It is only used as a placeholder, because
350
   * DirectAddressNameResolverProvider will not actually try to use it. However, it must be a valid
351
   * URI.
352
   */
353
  @VisibleForTesting
354
  static String makeTargetStringForDirectAddress(SocketAddress address) {
355
    try {
356
      return new URI(DIRECT_ADDRESS_SCHEME, "", "/" + address, null).toString();
1✔
357
    } catch (URISyntaxException e) {
×
358
      // It should not happen.
359
      throw new RuntimeException(e);
×
360
    }
361
  }
362

363
  /**
364
   * Creates a new managed channel builder with the given server address, authority string of the
365
   * channel. Transport implementors must provide client transport factory builder, and may set
366
   * custom channel default port provider.
367
   */
368
  public ManagedChannelImplBuilder(SocketAddress directServerAddress, String authority,
369
      ClientTransportFactoryBuilder clientTransportFactoryBuilder,
370
      @Nullable ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider) {
371
    this(directServerAddress, authority, null, null, clientTransportFactoryBuilder,
1✔
372
        channelBuilderDefaultPortProvider);
373
  }
1✔
374

375
  /**
376
   * Creates a new managed channel builder with the given server address, authority string of the
377
   * channel. Transport implementors must provide client transport factory builder, and may set
378
   * custom channel default port provider.
379
   * 
380
   * @param channelCreds The ChannelCredentials provided by the user. These may be used when
381
   *     creating derivative channels.
382
   */
383
  public ManagedChannelImplBuilder(SocketAddress directServerAddress, String authority,
384
      @Nullable ChannelCredentials channelCreds, @Nullable CallCredentials callCreds,
385
      ClientTransportFactoryBuilder clientTransportFactoryBuilder,
386
      @Nullable ChannelBuilderDefaultPortProvider channelBuilderDefaultPortProvider) {
1✔
387
    this.target = makeTargetStringForDirectAddress(directServerAddress);
1✔
388
    this.channelCredentials = channelCreds;
1✔
389
    this.callCredentials = callCreds;
1✔
390
    this.clientTransportFactoryBuilder = checkNotNull(clientTransportFactoryBuilder,
1✔
391
        "clientTransportFactoryBuilder");
392
    this.directServerAddress = directServerAddress;
1✔
393
    NameResolverRegistry reg = new NameResolverRegistry();
1✔
394
    reg.register(new DirectAddressNameResolverProvider(directServerAddress,
1✔
395
        authority));
396
    this.nameResolverRegistry = reg;
1✔
397

398
    if (channelBuilderDefaultPortProvider != null) {
1✔
399
      this.channelBuilderDefaultPortProvider = channelBuilderDefaultPortProvider;
1✔
400
    } else {
401
      this.channelBuilderDefaultPortProvider = new ManagedChannelDefaultPortProvider();
1✔
402
    }
403
    // TODO(dnvindhya): Move configurator to all the individual builders
404
    InternalConfiguratorRegistry.configureChannelBuilder(this);
1✔
405
  }
1✔
406

407
  @Override
408
  public ManagedChannelImplBuilder directExecutor() {
409
    return executor(MoreExecutors.directExecutor());
1✔
410
  }
411

412
  @Override
413
  public ManagedChannelImplBuilder executor(Executor executor) {
414
    if (executor != null) {
1✔
415
      this.executorPool = new FixedObjectPool<>(executor);
1✔
416
    } else {
417
      this.executorPool = DEFAULT_EXECUTOR_POOL;
1✔
418
    }
419
    return this;
1✔
420
  }
421

422
  @Override
423
  public ManagedChannelImplBuilder offloadExecutor(Executor executor) {
424
    if (executor != null) {
1✔
425
      this.offloadExecutorPool = new FixedObjectPool<>(executor);
1✔
426
    } else {
427
      this.offloadExecutorPool = DEFAULT_EXECUTOR_POOL;
1✔
428
    }
429
    return this;
1✔
430
  }
431

432
  @Override
433
  public ManagedChannelImplBuilder intercept(List<ClientInterceptor> interceptors) {
434
    this.interceptors.addAll(interceptors);
1✔
435
    return this;
1✔
436
  }
437

438
  @Override
439
  public ManagedChannelImplBuilder intercept(ClientInterceptor... interceptors) {
440
    return intercept(Arrays.asList(interceptors));
1✔
441
  }
442

443
  @Override
444
  protected ManagedChannelImplBuilder interceptWithTarget(InterceptorFactory factory) {
445
    // Add a placeholder instance to the interceptor list, and replace it with a real instance
446
    // during build().
447
    this.interceptors.add(new InterceptorFactoryWrapper(factory));
1✔
448
    return this;
1✔
449
  }
450

451
  @Override
452
  public ManagedChannelImplBuilder addTransportFilter(ClientTransportFilter hook) {
453
    transportFilters.add(checkNotNull(hook, "transport filter"));
1✔
454
    return this;
1✔
455
  }
456

457
  @Deprecated
458
  @Override
459
  public ManagedChannelImplBuilder nameResolverFactory(NameResolver.Factory resolverFactory) {
460
    Preconditions.checkState(directServerAddress == null,
1✔
461
        "directServerAddress is set (%s), which forbids the use of NameResolverFactory",
462
        directServerAddress);
463

464
    if (resolverFactory != null) {
1✔
465
      NameResolverRegistry reg = new NameResolverRegistry();
1✔
466
      if (resolverFactory instanceof NameResolverProvider) {
1✔
467
        reg.register((NameResolverProvider) resolverFactory);
1✔
468
      } else {
469
        reg.register(new NameResolverFactoryToProviderFacade(resolverFactory));
1✔
470
      }
471
      this.nameResolverRegistry = reg;
1✔
472
    } else {
1✔
473
      this.nameResolverRegistry = NameResolverRegistry.getDefaultRegistry();
1✔
474
    }
475
    return this;
1✔
476
  }
477

478
  ManagedChannelImplBuilder nameResolverRegistry(NameResolverRegistry resolverRegistry) {
479
    this.nameResolverRegistry = resolverRegistry;
1✔
480
    return this;
1✔
481
  }
482

483
  @Override
484
  public ManagedChannelImplBuilder defaultLoadBalancingPolicy(String policy) {
485
    Preconditions.checkState(directServerAddress == null,
1✔
486
        "directServerAddress is set (%s), which forbids the use of load-balancing policy",
487
        directServerAddress);
488
    Preconditions.checkArgument(policy != null, "policy cannot be null");
1✔
489
    this.defaultLbPolicy = policy;
1✔
490
    return this;
1✔
491
  }
492

493
  @Override
494
  public ManagedChannelImplBuilder decompressorRegistry(DecompressorRegistry registry) {
495
    if (registry != null) {
1✔
496
      this.decompressorRegistry = registry;
1✔
497
    } else {
498
      this.decompressorRegistry = DEFAULT_DECOMPRESSOR_REGISTRY;
1✔
499
    }
500
    return this;
1✔
501
  }
502

503
  @Override
504
  public ManagedChannelImplBuilder compressorRegistry(CompressorRegistry registry) {
505
    if (registry != null) {
1✔
506
      this.compressorRegistry = registry;
1✔
507
    } else {
508
      this.compressorRegistry = DEFAULT_COMPRESSOR_REGISTRY;
1✔
509
    }
510
    return this;
1✔
511
  }
512

513
  @Override
514
  public ManagedChannelImplBuilder userAgent(@Nullable String userAgent) {
515
    this.userAgent = userAgent;
1✔
516
    return this;
1✔
517
  }
518

519
  @Override
520
  public ManagedChannelImplBuilder overrideAuthority(String authority) {
521
    this.authorityOverride = checkAuthority(authority);
1✔
522
    return this;
1✔
523
  }
524

525
  @Override
526
  public ManagedChannelImplBuilder idleTimeout(long value, TimeUnit unit) {
527
    checkArgument(value > 0, "idle timeout is %s, but must be positive", value);
1✔
528
    // We convert to the largest unit to avoid overflow
529
    if (unit.toDays(value) >= IDLE_MODE_MAX_TIMEOUT_DAYS) {
1✔
530
      // This disables idle mode
531
      this.idleTimeoutMillis = ManagedChannelImpl.IDLE_TIMEOUT_MILLIS_DISABLE;
1✔
532
    } else {
533
      this.idleTimeoutMillis = Math.max(unit.toMillis(value), IDLE_MODE_MIN_TIMEOUT_MILLIS);
1✔
534
    }
535
    return this;
1✔
536
  }
537

538
  @Override
539
  public ManagedChannelImplBuilder maxRetryAttempts(int maxRetryAttempts) {
540
    this.maxRetryAttempts = maxRetryAttempts;
1✔
541
    return this;
1✔
542
  }
543

544
  @Override
545
  public ManagedChannelImplBuilder maxHedgedAttempts(int maxHedgedAttempts) {
546
    this.maxHedgedAttempts = maxHedgedAttempts;
1✔
547
    return this;
1✔
548
  }
549

550
  @Override
551
  public ManagedChannelImplBuilder retryBufferSize(long bytes) {
552
    checkArgument(bytes > 0L, "retry buffer size must be positive");
1✔
553
    retryBufferSize = bytes;
1✔
554
    return this;
1✔
555
  }
556

557
  @Override
558
  public ManagedChannelImplBuilder perRpcBufferLimit(long bytes) {
559
    checkArgument(bytes > 0L, "per RPC buffer limit must be positive");
1✔
560
    perRpcBufferLimit = bytes;
1✔
561
    return this;
1✔
562
  }
563

564
  @Override
565
  public ManagedChannelImplBuilder disableRetry() {
566
    retryEnabled = false;
1✔
567
    return this;
1✔
568
  }
569

570
  @Override
571
  public ManagedChannelImplBuilder enableRetry() {
572
    retryEnabled = true;
1✔
573
    return this;
1✔
574
  }
575

576
  @Override
577
  public ManagedChannelImplBuilder setBinaryLog(BinaryLog binlog) {
578
    this.binlog = binlog;
×
579
    return this;
×
580
  }
581

582
  @Override
583
  public ManagedChannelImplBuilder maxTraceEvents(int maxTraceEvents) {
584
    checkArgument(maxTraceEvents >= 0, "maxTraceEvents must be non-negative");
1✔
585
    this.maxTraceEvents = maxTraceEvents;
1✔
586
    return this;
1✔
587
  }
588

589
  @Override
590
  public ManagedChannelImplBuilder proxyDetector(@Nullable ProxyDetector proxyDetector) {
591
    this.proxyDetector = proxyDetector;
1✔
592
    return this;
1✔
593
  }
594

595
  @Override
596
  public ManagedChannelImplBuilder defaultServiceConfig(@Nullable Map<String, ?> serviceConfig) {
597
    // TODO(notcarl): use real parsing
598
    defaultServiceConfig = checkMapEntryTypes(serviceConfig);
1✔
599
    return this;
1✔
600
  }
601

602
  @Nullable
603
  private static Map<String, ?> checkMapEntryTypes(@Nullable Map<?, ?> map) {
604
    if (map == null) {
1✔
605
      return null;
×
606
    }
607
    // Not using ImmutableMap.Builder because of extra guava dependency for Android.
608
    Map<String, Object> parsedMap = new LinkedHashMap<>();
1✔
609
    for (Map.Entry<?, ?> entry : map.entrySet()) {
1✔
610
      checkArgument(
1✔
611
          entry.getKey() instanceof String,
1✔
612
          "The key of the entry '%s' is not of String type", entry);
613

614
      String key = (String) entry.getKey();
1✔
615
      Object value = entry.getValue();
1✔
616
      if (value == null) {
1✔
617
        parsedMap.put(key, null);
1✔
618
      } else if (value instanceof Map) {
1✔
619
        parsedMap.put(key, checkMapEntryTypes((Map<?, ?>) value));
1✔
620
      } else if (value instanceof List) {
1✔
621
        parsedMap.put(key, checkListEntryTypes((List<?>) value));
1✔
622
      } else if (value instanceof String) {
1✔
623
        parsedMap.put(key, value);
1✔
624
      } else if (value instanceof Number) {
1✔
625
        parsedMap.put(key, ((Number) value).doubleValue());
1✔
626
      } else if (value instanceof Boolean) {
1✔
627
        parsedMap.put(key, value);
1✔
628
      } else {
629
        throw new IllegalArgumentException(
×
630
            "The value of the map entry '" + entry + "' is of type '" + value.getClass()
×
631
                + "', which is not supported");
632
      }
633
    }
1✔
634
    return Collections.unmodifiableMap(parsedMap);
1✔
635
  }
636

637
  private static List<?> checkListEntryTypes(List<?> list) {
638
    List<Object> parsedList = new ArrayList<>(list.size());
1✔
639
    for (Object value : list) {
1✔
640
      if (value == null) {
1✔
641
        parsedList.add(null);
1✔
642
      } else if (value instanceof Map) {
1✔
643
        parsedList.add(checkMapEntryTypes((Map<?, ?>) value));
1✔
644
      } else if (value instanceof List) {
1✔
645
        parsedList.add(checkListEntryTypes((List<?>) value));
×
646
      } else if (value instanceof String) {
1✔
647
        parsedList.add(value);
1✔
648
      } else if (value instanceof Number) {
1✔
649
        parsedList.add(((Number) value).doubleValue());
1✔
650
      } else if (value instanceof Boolean) {
1✔
651
        parsedList.add(value);
1✔
652
      } else {
653
        throw new IllegalArgumentException(
×
654
            "The entry '" + value + "' is of type '" + value.getClass()
×
655
                + "', which is not supported");
656
      }
657
    }
1✔
658
    return Collections.unmodifiableList(parsedList);
1✔
659
  }
660

661
  @Override
662
  public <X> ManagedChannelImplBuilder setNameResolverArg(NameResolver.Args.Key<X> key, X value) {
663
    if (nameResolverCustomArgs == null) {
1✔
664
      nameResolverCustomArgs = new IdentityHashMap<>();
1✔
665
    }
666
    nameResolverCustomArgs.put(checkNotNull(key, "key"), checkNotNull(value, "value"));
1✔
667
    return this;
1✔
668
  }
669

670
  @SuppressWarnings("unchecked") // This cast is safe because of setNameResolverArg()'s signature.
671
  void copyAllNameResolverCustomArgsTo(NameResolver.Args.Builder dest) {
672
    if (nameResolverCustomArgs != null) {
1✔
673
      for (Map.Entry<NameResolver.Args.Key<?>, Object> entry : nameResolverCustomArgs.entrySet()) {
1✔
674
        dest.setArg((NameResolver.Args.Key<Object>) entry.getKey(), entry.getValue());
1✔
675
      }
1✔
676
    }
677
  }
1✔
678

679
  @Override
680
  public ManagedChannelImplBuilder disableServiceConfigLookUp() {
681
    this.lookUpServiceConfig = false;
1✔
682
    return this;
1✔
683
  }
684

685
  /**
686
   * Disable or enable stats features. Enabled by default.
687
   *
688
   * <p>For the current release, calling {@code setStatsEnabled(true)} may have a side effect that
689
   * disables retry.
690
   */
691
  public void setStatsEnabled(boolean value) {
692
    statsEnabled = value;
1✔
693
  }
1✔
694

695
  /**
696
   * Disable or enable stats recording for RPC upstarts.  Effective only if {@link
697
   * #setStatsEnabled} is set to true.  Enabled by default.
698
   */
699
  public void setStatsRecordStartedRpcs(boolean value) {
700
    recordStartedRpcs = value;
1✔
701
  }
1✔
702

703
  /**
704
   * Disable or enable stats recording for RPC completions.  Effective only if {@link
705
   * #setStatsEnabled} is set to true.  Enabled by default.
706
   */
707
  public void setStatsRecordFinishedRpcs(boolean value) {
708
    recordFinishedRpcs = value;
1✔
709
  }
1✔
710

711
  /**
712
   * Disable or enable real-time metrics recording.  Effective only if {@link #setStatsEnabled} is
713
   * set to true.  Disabled by default.
714
   */
715
  public void setStatsRecordRealTimeMetrics(boolean value) {
716
    recordRealTimeMetrics = value;
×
717
  }
×
718
  
719
  public void setStatsRecordRetryMetrics(boolean value) {
720
    recordRetryMetrics = value;
1✔
721
  }
1✔
722

723
  /**
724
   * Disable or enable tracing features.  Enabled by default.
725
   */
726
  public void setTracingEnabled(boolean value) {
727
    tracingEnabled = value;
1✔
728
  }
1✔
729

730
  /**
731
   * Verifies the authority is valid.
732
   */
733
  @VisibleForTesting
734
  String checkAuthority(String authority) {
735
    if (authorityCheckerDisabled) {
1✔
736
      return authority;
1✔
737
    }
738
    return GrpcUtil.checkAuthority(authority);
1✔
739
  }
740

741
  /** Disable the check whether the authority is valid. */
742
  public ManagedChannelImplBuilder disableCheckAuthority() {
743
    authorityCheckerDisabled = true;
1✔
744
    return this;
1✔
745
  }
746

747
  /** Enable previously disabled authority check. */
748
  public ManagedChannelImplBuilder enableCheckAuthority() {
749
    authorityCheckerDisabled = false;
1✔
750
    return this;
1✔
751
  }
752

753
  @Override
754
  protected ManagedChannelImplBuilder addMetricSink(MetricSink metricSink) {
755
    metricSinks.add(checkNotNull(metricSink, "metric sink"));
1✔
756
    return this;
1✔
757
  }
758

759
  @Override
760
  public ManagedChannel build() {
761
    ClientTransportFactory clientTransportFactory =
1✔
762
        clientTransportFactoryBuilder.buildClientTransportFactory();
1✔
763
    ResolvedNameResolver resolvedResolver =
764
        InternalFeatureFlags.getRfc3986UrisEnabled()
1✔
765
            ? getNameResolverProviderRfc3986(target, nameResolverRegistry, nameResolverProvider)
1✔
766
            : getNameResolverProvider(target, nameResolverRegistry, nameResolverProvider);
1✔
767
    resolvedResolver.checkAddressTypes(clientTransportFactory.getSupportedSocketAddressTypes());
1✔
768
    return new ManagedChannelOrphanWrapper(new ManagedChannelImpl(
1✔
769
        this,
770
        clientTransportFactory,
771
        resolvedResolver.targetUri,
772
        resolvedResolver.provider,
773
        new ExponentialBackoffPolicy.Provider(),
774
        SharedResourcePool.forResource(GrpcUtil.SHARED_CHANNEL_EXECUTOR),
1✔
775
        GrpcUtil.STOPWATCH_SUPPLIER,
776
        getEffectiveInterceptors(resolvedResolver.targetUri.toString()),
1✔
777
        TimeProvider.SYSTEM_TIME_PROVIDER));
778
  }
779

780
  // Temporarily disable retry when stats or tracing is enabled to avoid breakage, until we know
781
  // what should be the desired behavior for retry + stats/tracing.
782
  // TODO(zdapeng): FIX IT
783
  @VisibleForTesting
784
  List<ClientInterceptor> getEffectiveInterceptors(String computedTarget) {
785
    List<ClientInterceptor> effectiveInterceptors = new ArrayList<>(this.interceptors.size());
1✔
786
    for (ClientInterceptor interceptor : this.interceptors) {
1✔
787
      if (interceptor instanceof InterceptorFactoryWrapper) {
1✔
788
        InterceptorFactory factory = ((InterceptorFactoryWrapper) interceptor).factory;
1✔
789
        interceptor = factory.newInterceptor(computedTarget);
1✔
790
        if (interceptor == null) {
1✔
791
          throw new NullPointerException("Factory returned null interceptor: " + factory);
×
792
        }
793
      }
794
      effectiveInterceptors.add(interceptor);
1✔
795
    }
1✔
796

797
    boolean disableImplicitCensus = InternalConfiguratorRegistry.wasSetConfiguratorsCalled();
1✔
798
    if (disableImplicitCensus) {
1✔
799
      return effectiveInterceptors;
×
800
    }
801
    if (statsEnabled) {
1✔
802
      ClientInterceptor statsInterceptor = null;
1✔
803

804
      if (GET_CLIENT_INTERCEPTOR_METHOD != null) {
1✔
805
        try {
806
          statsInterceptor =
1✔
807
              (ClientInterceptor) GET_CLIENT_INTERCEPTOR_METHOD
808
              .invoke(
1✔
809
                null,
810
                recordStartedRpcs,
1✔
811
                recordFinishedRpcs,
1✔
812
                recordRealTimeMetrics,
1✔
813
                recordRetryMetrics);
1✔
814
        } catch (IllegalAccessException e) {
×
815
          log.log(Level.FINE, "Unable to apply census stats", e);
×
816
        } catch (InvocationTargetException e) {
×
817
          log.log(Level.FINE, "Unable to apply census stats", e);
×
818
        }
1✔
819
      }
820

821
      if (statsInterceptor != null) {
1✔
822
        // First interceptor runs last (see ClientInterceptors.intercept()), so that no
823
        // other interceptor can override the tracer factory we set in CallOptions.
824
        effectiveInterceptors.add(0, statsInterceptor);
1✔
825
      }
826
    }
827
    if (tracingEnabled) {
1✔
828
      ClientInterceptor tracingInterceptor = null;
1✔
829
      try {
830
        Class<?> censusTracingAccessor =
1✔
831
            Class.forName("io.grpc.census.InternalCensusTracingAccessor");
1✔
832
        Method getClientInterceptroMethod =
1✔
833
            censusTracingAccessor.getDeclaredMethod("getClientInterceptor");
1✔
834
        tracingInterceptor = (ClientInterceptor) getClientInterceptroMethod.invoke(null);
1✔
835
      } catch (ClassNotFoundException e) {
1✔
836
        // Replace these separate catch statements with multicatch when Android min-API >= 19
837
        log.log(Level.FINE, "Unable to apply census stats", e);
1✔
838
      } catch (NoSuchMethodException e) {
×
839
        log.log(Level.FINE, "Unable to apply census stats", e);
×
840
      } catch (IllegalAccessException e) {
×
841
        log.log(Level.FINE, "Unable to apply census stats", e);
×
842
      } catch (InvocationTargetException e) {
×
843
        log.log(Level.FINE, "Unable to apply census stats", e);
×
844
      }
1✔
845
      if (tracingInterceptor != null) {
1✔
846
        effectiveInterceptors.add(0, tracingInterceptor);
1✔
847
      }
848
    }
849
    return effectiveInterceptors;
1✔
850
  }
851

852
  /**
853
   * Returns a default port to {@link NameResolver} for use in cases where the target string doesn't
854
   * include a port. The default implementation returns {@link GrpcUtil#DEFAULT_PORT_SSL}.
855
   */
856
  int getDefaultPort() {
857
    return channelBuilderDefaultPortProvider.getDefaultPort();
1✔
858
  }
859

860
  @VisibleForTesting
861
  static class ResolvedNameResolver {
862
    public final UriWrapper targetUri;
863
    public final NameResolverProvider provider;
864

865
    public ResolvedNameResolver(UriWrapper targetUri, NameResolverProvider provider) {
1✔
866
      this.targetUri = checkNotNull(targetUri, "targetUri");
1✔
867
      this.provider = checkNotNull(provider, "provider");
1✔
868
    }
1✔
869

870
    void checkAddressTypes(
871
        Collection<Class<? extends SocketAddress>> channelTransportSocketAddressTypes) {
872
      if (channelTransportSocketAddressTypes != null) {
1✔
873
        Collection<Class<? extends SocketAddress>> nameResolverSocketAddressTypes =
1✔
874
            provider.getProducedSocketAddressTypes();
1✔
875
        if (!channelTransportSocketAddressTypes.containsAll(nameResolverSocketAddressTypes)) {
1✔
876
          throw new IllegalArgumentException(
1✔
877
              String.format(
1✔
878
                  "Address types of NameResolver '%s' for '%s' not supported by transport",
879
                  provider.getDefaultScheme(), targetUri));
1✔
880
        }
881
      }
882
    }
1✔
883
  }
884

885
  @VisibleForTesting
886
  static ResolvedNameResolver getNameResolverProvider(
887
      String target, NameResolverRegistry nameResolverRegistry,
888
      NameResolverProvider nameResolverProvider) {
889
    // Finding a NameResolver. Try using the target string as the URI. If that fails, try prepending
890
    // "dns:///".
891
    NameResolverProvider provider = null;
1✔
892
    URI targetUri = null;
1✔
893
    StringBuilder uriSyntaxErrors = new StringBuilder();
1✔
894
    try {
895
      targetUri = new URI(target);
1✔
896
    } catch (URISyntaxException e) {
1✔
897
      // Can happen with ip addresses like "[::1]:1234" or 127.0.0.1:1234.
898
      uriSyntaxErrors.append(e.getMessage());
1✔
899
    }
1✔
900
    if (targetUri != null) {
1✔
901
      // For "localhost:8080" this would likely cause provider to be null, because "localhost" is
902
      // parsed as the scheme. Will hit the next case and try "dns:///localhost:8080".
903
      // Use the explicit provider if its scheme matches the target URI.
904
      if (nameResolverProvider != null
1✔
905
          && targetUri.getScheme().equals(nameResolverProvider.getScheme())) {
1✔
906
        provider = nameResolverProvider;
1✔
907
      } else {
908
        provider = nameResolverRegistry.getProviderForScheme(targetUri.getScheme());
1✔
909
      }
910
    }
911

912
    if (provider == null && !URI_PATTERN.matcher(target).matches()) {
1✔
913
      // It doesn't look like a URI target. Maybe it's an authority string. Try with
914
      // the default scheme from the registry (if provider is not specified) or
915
      // the provider's default scheme (if provider is specified).
916
      String scheme = nameResolverProvider != null
1✔
917
          ? nameResolverProvider.getScheme()
1✔
918
          : nameResolverRegistry.getDefaultScheme();
1✔
919
      try {
920
        targetUri = new URI(scheme, "", "/" + target, null);
1✔
921
      } catch (URISyntaxException e) {
×
922
        // Should not be possible
923
        throw new IllegalArgumentException(e);
×
924
      }
1✔
925
      if (nameResolverProvider != null) {
1✔
926
        provider = nameResolverProvider;
1✔
927
      } else {
928
        provider = nameResolverRegistry.getProviderForScheme(targetUri.getScheme());
1✔
929
      }
930
    }
931

932
    if (provider == null) {
1✔
933
      throw new IllegalArgumentException(String.format(
1✔
934
          "Could not find a NameResolverProvider for %s%s",
935
          target, uriSyntaxErrors.length() > 0 ? " (" + uriSyntaxErrors + ")" : ""));
1✔
936
    }
937

938
    return new ResolvedNameResolver(wrap(targetUri), provider);
1✔
939
  }
940

941
  @VisibleForTesting
942
  static ResolvedNameResolver getNameResolverProviderRfc3986(
943
      String target, NameResolverRegistry nameResolverRegistry,
944
      NameResolverProvider nameResolverProvider) {
945
    // Finding a NameResolver. Try using the target string as the URI. If that fails, try prepending
946
    // "dns:///".
947
    NameResolverProvider provider = null;
1✔
948
    Uri targetUri = null;
1✔
949
    StringBuilder uriSyntaxErrors = new StringBuilder();
1✔
950
    try {
951
      targetUri = Uri.parse(target);
1✔
952
    } catch (URISyntaxException e) {
1✔
953
      // Can happen with ip addresses like "[::1]:1234" or 127.0.0.1:1234.
954
      uriSyntaxErrors.append(e.getMessage());
1✔
955
    }
1✔
956
    if (targetUri != null) {
1✔
957
      // For "localhost:8080" this would likely cause provider to be null, because "localhost" is
958
      // parsed as the scheme. Will hit the next case and try "dns:///localhost:8080".
959
      // Use the explicit provider if its scheme matches the target URI.
960
      if (nameResolverProvider != null
1✔
961
          && targetUri.getScheme().equals(nameResolverProvider.getScheme())) {
1✔
962
        provider = nameResolverProvider;
1✔
963
      } else {
964
        provider = nameResolverRegistry.getProviderForScheme(targetUri.getScheme());
1✔
965
      }
966
    }
967

968
    if (provider == null && !URI_PATTERN.matcher(target).matches()) {
1✔
969
      // It doesn't look like a URI target. Maybe it's an authority string. Try with
970
      // the default scheme from the registry (if provider is not specified) or
971
      // the provider's default scheme (if provider is specified).
972
      String scheme = nameResolverProvider != null
1✔
973
          ? nameResolverProvider.getScheme()
1✔
974
          : nameResolverRegistry.getDefaultScheme();
1✔
975
      targetUri =
976
          Uri.newBuilder()
1✔
977
              .setScheme(scheme)
1✔
978
              .setHost("")
1✔
979
              .setPath("/" + target)
1✔
980
              .build();
1✔
981
      provider = nameResolverRegistry.getProviderForScheme(targetUri.getScheme());
1✔
982
    }
983

984
    if (provider == null) {
1✔
985
      throw new IllegalArgumentException(
1✔
986
          String.format(
1✔
987
              "Could not find a NameResolverProvider for %s%s",
988
              target, uriSyntaxErrors.length() > 0 ? " (" + uriSyntaxErrors + ")" : ""));
1✔
989
    }
990

991
    return new ResolvedNameResolver(wrap(targetUri), provider);
1✔
992
  }
993

994
  private static class DirectAddressNameResolverProvider extends NameResolverProvider {
995
    final SocketAddress address;
996
    final String authority;
997
    final Collection<Class<? extends SocketAddress>> producedSocketAddressTypes;
998

999
    DirectAddressNameResolverProvider(SocketAddress address, String authority) {
1✔
1000
      this.address = address;
1✔
1001
      this.authority = authority;
1✔
1002
      this.producedSocketAddressTypes
1✔
1003
          = Collections.singleton(address.getClass());
1✔
1004
    }
1✔
1005

1006
    @Override
1007
    public NameResolver newNameResolver(URI notUsedUri, NameResolver.Args args) {
1008
      return new NameResolver() {
1✔
1009
        @Override
1010
        public String getServiceAuthority() {
1011
          return authority;
1✔
1012
        }
1013

1014
        @Override
1015
        public void start(Listener2 listener) {
1016
          listener.onResult2(
1✔
1017
              ResolutionResult.newBuilder()
1✔
1018
                  .setAddressesOrError(
1✔
1019
                      StatusOr.fromValue(
1✔
1020
                          Collections.singletonList(new EquivalentAddressGroup(address))))
1✔
1021
                  .setAttributes(Attributes.EMPTY)
1✔
1022
                  .build());
1✔
1023
        }
1✔
1024

1025
        @Override
1026
        public void shutdown() {}
1✔
1027
      };
1028
    }
1029

1030
    @Override
1031
    public String getDefaultScheme() {
1032
      return DIRECT_ADDRESS_SCHEME;
1✔
1033
    }
1034

1035
    @Override
1036
    protected boolean isAvailable() {
1037
      return true;
1✔
1038
    }
1039

1040
    @Override
1041
    protected int priority() {
1042
      return 5;
1✔
1043
    }
1044

1045
    @Override
1046
    public Collection<Class<? extends SocketAddress>> getProducedSocketAddressTypes() {
1047
      return producedSocketAddressTypes;
1✔
1048
    }
1049
  }
1050

1051
  private static final class InterceptorFactoryWrapper implements ClientInterceptor {
1052
    final InterceptorFactory factory;
1053

1054
    public InterceptorFactoryWrapper(InterceptorFactory factory) {
1✔
1055
      this.factory = checkNotNull(factory, "factory");
1✔
1056
    }
1✔
1057

1058
    @Override
1059
    public <ReqT, RespT> ClientCall<ReqT, RespT> interceptCall(
1060
        MethodDescriptor<ReqT, RespT> method, CallOptions callOptions, Channel next) {
1061
      throw new AssertionError("Should have been replaced with real instance");
×
1062
    }
1063
  }
1064

1065
  /**
1066
   * Returns the internal offload executor pool for offloading tasks.
1067
   */
1068
  public ObjectPool<? extends Executor> getOffloadExecutorPool() {
1069
    return this.offloadExecutorPool;
1✔
1070
  }
1071
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc