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

grpc / grpc-java / #19970

29 Aug 2025 09:01PM UTC coverage: 88.532% (-0.06%) from 88.588%
#19970

push

github

web-flow
Upgrade guava version to 33.4.8 (#12219)

Guava seems to call a deprecated sun.misc.Unsafe::objectFieldOffset
method which might be removed in a future JDK release.
There are still a few things to do for -android versions, which are
tracked in https://github.com/google/guava/issues/7742,
see details at google/guava#7811

Fixes #12215

34678 of 39170 relevant lines covered (88.53%)

0.89 hits per line

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

78.51
/../servlet/src/main/java/io/grpc/servlet/ServletAdapter.java
1
/*
2
 * Copyright 2018 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.servlet;
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.GrpcUtil.TIMEOUT_KEY;
22
import static java.util.logging.Level.FINE;
23
import static java.util.logging.Level.FINEST;
24

25
import com.google.common.io.BaseEncoding;
26
import io.grpc.Attributes;
27
import io.grpc.ExperimentalApi;
28
import io.grpc.Grpc;
29
import io.grpc.InternalLogId;
30
import io.grpc.InternalMetadata;
31
import io.grpc.Metadata;
32
import io.grpc.ServerStreamTracer;
33
import io.grpc.Status;
34
import io.grpc.internal.GrpcUtil;
35
import io.grpc.internal.ReadableBuffers;
36
import io.grpc.internal.ServerTransportListener;
37
import io.grpc.internal.StatsTraceContext;
38
import java.io.IOException;
39
import java.net.InetSocketAddress;
40
import java.net.URI;
41
import java.net.URISyntaxException;
42
import java.nio.charset.StandardCharsets;
43
import java.util.ArrayList;
44
import java.util.Arrays;
45
import java.util.Enumeration;
46
import java.util.List;
47
import java.util.concurrent.TimeUnit;
48
import java.util.logging.Logger;
49
import javax.servlet.AsyncContext;
50
import javax.servlet.AsyncEvent;
51
import javax.servlet.AsyncListener;
52
import javax.servlet.ReadListener;
53
import javax.servlet.ServletInputStream;
54
import javax.servlet.http.HttpServletRequest;
55
import javax.servlet.http.HttpServletResponse;
56

57
/**
58
 * An adapter that transforms {@link HttpServletRequest} into gRPC request and lets a gRPC server
59
 * process it, and transforms the gRPC response into {@link HttpServletResponse}. An adapter can be
60
 * instantiated by {@link ServletServerBuilder#buildServletAdapter()}.
61
 *
62
 * <p>In a servlet, calling {@link #doPost(HttpServletRequest, HttpServletResponse)} inside {@link
63
 * javax.servlet.http.HttpServlet#doPost(HttpServletRequest, HttpServletResponse)} makes the servlet
64
 * backed by the gRPC server associated with the adapter. The servlet must support Asynchronous
65
 * Processing and must be deployed to a container that supports servlet 4.0 and enables HTTP/2.
66
 *
67
 * <p>The API is experimental. The authors would like to know more about the real usecases. Users
68
 * are welcome to provide feedback by commenting on
69
 * <a href=https://github.com/grpc/grpc-java/issues/5066>the tracking issue</a>.
70
 */
71
@ExperimentalApi("https://github.com/grpc/grpc-java/issues/5066")
72
public final class ServletAdapter {
73

74
  static final Logger logger = Logger.getLogger(ServletAdapter.class.getName());
1✔
75

76
  private final ServerTransportListener transportListener;
77
  private final List<? extends ServerStreamTracer.Factory> streamTracerFactories;
78
  private final int maxInboundMessageSize;
79
  private final Attributes attributes;
80

81
  ServletAdapter(
82
      ServerTransportListener transportListener,
83
      List<? extends ServerStreamTracer.Factory> streamTracerFactories,
84
      int maxInboundMessageSize) {
1✔
85
    this.transportListener = transportListener;
1✔
86
    this.streamTracerFactories = streamTracerFactories;
1✔
87
    this.maxInboundMessageSize = maxInboundMessageSize;
1✔
88
    attributes = transportListener.transportReady(Attributes.EMPTY);
1✔
89
  }
1✔
90

91
  /**
92
   * Call this method inside {@link javax.servlet.http.HttpServlet#doGet(HttpServletRequest,
93
   * HttpServletResponse)} to serve gRPC GET request.
94
   *
95
   * <p>This method is currently not implemented.
96
   *
97
   * <p>Note that in rare case gRPC client sends GET requests.
98
   *
99
   * <p>Do not modify {@code req} and {@code resp} before or after calling this method. However,
100
   * calling {@code resp.setBufferSize()} before invocation is allowed.
101
   */
102
  public void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException {
103
    resp.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "GET method not supported");
×
104
  }
×
105

106
  /**
107
   * Call this method inside {@link javax.servlet.http.HttpServlet#doPost(HttpServletRequest,
108
   * HttpServletResponse)} to serve gRPC POST request.
109
   *
110
   * <p>Do not modify {@code req} and {@code resp} before or after calling this method. However,
111
   * calling {@code resp.setBufferSize()} before invocation is allowed.
112
   */
113
  public void doPost(HttpServletRequest req, HttpServletResponse resp) throws IOException {
114
    checkArgument(req.isAsyncSupported(), "servlet does not support asynchronous operation");
1✔
115
    checkArgument(ServletAdapter.isGrpc(req), "the request is not a gRPC request");
1✔
116

117
    InternalLogId logId = InternalLogId.allocate(ServletAdapter.class, null);
1✔
118
    logger.log(FINE, "[{0}] RPC started", logId);
1✔
119

120
    AsyncContext asyncCtx = req.startAsync(req, resp);
1✔
121

122
    String method = req.getRequestURI().substring(1); // remove the leading "/"
1✔
123
    Metadata headers = getHeaders(req);
1✔
124

125
    if (logger.isLoggable(FINEST)) {
1✔
126
      logger.log(FINEST, "[{0}] method: {1}", new Object[] {logId, method});
×
127
      logger.log(FINEST, "[{0}] headers: {1}", new Object[] {logId, headers});
×
128
    }
129

130
    Long timeoutNanos = headers.get(TIMEOUT_KEY);
1✔
131
    if (timeoutNanos == null) {
1✔
132
      timeoutNanos = 0L;
1✔
133
    }
134
    asyncCtx.setTimeout(TimeUnit.NANOSECONDS.toMillis(timeoutNanos));
1✔
135
    StatsTraceContext statsTraceCtx =
1✔
136
        StatsTraceContext.newServerContext(streamTracerFactories, method, headers);
1✔
137

138
    ServletServerStream stream = new ServletServerStream(
1✔
139
        asyncCtx,
140
        statsTraceCtx,
141
        maxInboundMessageSize,
142
        attributes.toBuilder()
1✔
143
            .set(
1✔
144
                Grpc.TRANSPORT_ATTR_REMOTE_ADDR,
145
                new InetSocketAddress(req.getRemoteHost(), req.getRemotePort()))
1✔
146
            .set(
1✔
147
                Grpc.TRANSPORT_ATTR_LOCAL_ADDR,
148
                new InetSocketAddress(req.getLocalAddr(), req.getLocalPort()))
1✔
149
            .build(),
1✔
150
        getAuthority(req),
1✔
151
        logId);
152

153
    transportListener.streamCreated(stream, method, headers);
1✔
154
    stream.transportState().runOnTransportThread(stream.transportState()::onStreamAllocated);
1✔
155

156
    asyncCtx.getRequest().getInputStream()
1✔
157
        .setReadListener(new GrpcReadListener(stream, asyncCtx, logId));
1✔
158
    asyncCtx.addListener(new GrpcAsyncListener(stream, logId));
1✔
159
  }
1✔
160

161
  // This method must use Enumeration and its members, since that is the only way to read headers
162
  // from the servlet api.
163
  @SuppressWarnings("JdkObsolete")
164
  private static Metadata getHeaders(HttpServletRequest req) {
165
    Enumeration<String> headerNames = req.getHeaderNames();
1✔
166
    checkNotNull(
1✔
167
        headerNames, "Servlet container does not allow HttpServletRequest.getHeaderNames()");
168
    List<byte[]> byteArrays = new ArrayList<>();
1✔
169
    while (headerNames.hasMoreElements()) {
1✔
170
      String headerName = headerNames.nextElement();
1✔
171
      Enumeration<String> values = req.getHeaders(headerName);
1✔
172
      if (values == null) {
1✔
173
        continue;
×
174
      }
175
      while (values.hasMoreElements()) {
1✔
176
        String value = values.nextElement();
1✔
177
        if (headerName.endsWith(Metadata.BINARY_HEADER_SUFFIX)) {
1✔
178
          byteArrays.add(headerName.getBytes(StandardCharsets.US_ASCII));
1✔
179
          byteArrays.add(BaseEncoding.base64().decode(value));
1✔
180
        } else {
181
          byteArrays.add(headerName.getBytes(StandardCharsets.US_ASCII));
1✔
182
          byteArrays.add(value.getBytes(StandardCharsets.US_ASCII));
1✔
183
        }
184
      }
1✔
185
    }
1✔
186
    return InternalMetadata.newMetadata(byteArrays.toArray(new byte[][]{}));
1✔
187
  }
188

189
  // This method must use HttpRequest#getRequestURL or HttpUtils#getRequestURL, both of which
190
  // can only return StringBuffer instances
191
  @SuppressWarnings("JdkObsolete")
192
  private static String getAuthority(HttpServletRequest req) {
193
    try {
194
      return new URI(req.getRequestURL().toString()).getAuthority();
1✔
195
    } catch (URISyntaxException e) {
×
196
      logger.log(FINE, "Error getting authority from the request URL {0}", req.getRequestURL());
×
197
      return req.getServerName() + ":" + req.getServerPort();
×
198
    }
199
  }
200

201
  /**
202
   * Call this method when the adapter is no longer needed. The gRPC server will be terminated.
203
   */
204
  public void destroy() {
205
    transportListener.transportTerminated();
1✔
206
  }
1✔
207

208
  private static final class GrpcAsyncListener implements AsyncListener {
209
    final InternalLogId logId;
210
    final ServletServerStream stream;
211

212
    GrpcAsyncListener(ServletServerStream stream, InternalLogId logId) {
1✔
213
      this.stream = stream;
1✔
214
      this.logId = logId;
1✔
215
    }
1✔
216

217
    @Override
218
    public void onComplete(AsyncEvent event) {
219
      stream.asyncCompleted = true;
1✔
220
    }
1✔
221

222
    @Override
223
    public void onTimeout(AsyncEvent event) {
224
      if (logger.isLoggable(FINE)) {
1✔
225
        logger.log(FINE, String.format("[{%s}] Timeout: ", logId), event.getThrowable());
×
226
      }
227
      // If the resp is not committed, cancel() to avoid being redirected to an error page.
228
      // Else, the container will send RST_STREAM in the end.
229
      if (!event.getAsyncContext().getResponse().isCommitted()) {
1✔
230
        stream.cancel(Status.DEADLINE_EXCEEDED);
1✔
231
      } else {
232
        stream.transportState().runOnTransportThread(
1✔
233
            () -> stream.transportState().transportReportStatus(Status.DEADLINE_EXCEEDED));
1✔
234
      }
235
    }
1✔
236

237
    @Override
238
    public void onError(AsyncEvent event) {
239
      if (logger.isLoggable(FINE)) {
×
240
        logger.log(FINE, String.format("[{%s}] Error: ", logId), event.getThrowable());
×
241
      }
242

243
      // If the resp is not committed, cancel() to avoid being redirected to an error page.
244
      // Else, the container will send RST_STREAM at the end.
245
      if (!event.getAsyncContext().getResponse().isCommitted()) {
×
246
        stream.cancel(Status.fromThrowable(event.getThrowable()));
×
247
      } else {
248
        stream.transportState().runOnTransportThread(
×
249
            () -> stream.transportState().transportReportStatus(
×
250
                Status.fromThrowable(event.getThrowable())));
×
251
      }
252
    }
×
253

254
    @Override
255
    public void onStartAsync(AsyncEvent event) {}
×
256
  }
257

258
  private static final class GrpcReadListener implements ReadListener {
259
    final ServletServerStream stream;
260
    final AsyncContext asyncCtx;
261
    final ServletInputStream input;
262
    final InternalLogId logId;
263

264
    GrpcReadListener(
265
        ServletServerStream stream,
266
        AsyncContext asyncCtx,
267
        InternalLogId logId) throws IOException {
1✔
268
      this.stream = stream;
1✔
269
      this.asyncCtx = asyncCtx;
1✔
270
      input = asyncCtx.getRequest().getInputStream();
1✔
271
      this.logId = logId;
1✔
272
    }
1✔
273

274
    final byte[] buffer = new byte[4 * 1024];
1✔
275

276
    @Override
277
    public void onDataAvailable() throws IOException {
278
      logger.log(FINEST, "[{0}] onDataAvailable: ENTRY", logId);
1✔
279

280
      while (input.isReady()) {
1✔
281
        int length = input.read(buffer);
1✔
282
        if (length == -1) {
1✔
283
          logger.log(FINEST, "[{0}] inbound data: read end of stream", logId);
×
284
          return;
×
285
        } else {
286
          if (logger.isLoggable(FINEST)) {
1✔
287
            logger.log(
×
288
                FINEST,
289
                "[{0}] inbound data: length = {1}, bytes = {2}",
290
                new Object[] {logId, length, ServletServerStream.toHexString(buffer, length)});
×
291
          }
292

293
          byte[] copy = Arrays.copyOf(buffer, length);
1✔
294
          stream.transportState().runOnTransportThread(
1✔
295
              () -> stream.transportState().inboundDataReceived(ReadableBuffers.wrap(copy), false));
1✔
296
        }
297
      }
1✔
298

299
      logger.log(FINEST, "[{0}] onDataAvailable: EXIT", logId);
1✔
300
    }
1✔
301

302
    @Override
303
    public void onAllDataRead() {
304
      logger.log(FINE, "[{0}] onAllDataRead", logId);
1✔
305
      stream.transportState().runOnTransportThread(() ->
1✔
306
          stream.transportState().inboundDataReceived(ReadableBuffers.empty(), true));
1✔
307
    }
1✔
308

309
    @Override
310
    public void onError(Throwable t) {
311
      if (logger.isLoggable(FINE)) {
1✔
312
        logger.log(FINE, String.format("[{%s}] Error: ", logId), t);
×
313
      }
314
      // If the resp is not committed, cancel() to avoid being redirected to an error page.
315
      // Else, the container will send RST_STREAM at the end.
316
      if (!asyncCtx.getResponse().isCommitted()) {
1✔
317
        stream.cancel(Status.fromThrowable(t));
1✔
318
      } else {
319
        stream.transportState().runOnTransportThread(
×
320
            () -> stream.transportState()
×
321
                .transportReportStatus(Status.fromThrowable(t)));
×
322
      }
323
    }
1✔
324
  }
325

326
  /**
327
   * Checks whether an incoming {@code HttpServletRequest} may come from a gRPC client.
328
   *
329
   * @return true if the request comes from a gRPC client
330
   */
331
  public static boolean isGrpc(HttpServletRequest request) {
332
    return request.getContentType() != null
1✔
333
        && request.getContentType().contains(GrpcUtil.CONTENT_TYPE_GRPC);
1✔
334
  }
335
}
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

© 2025 Coveralls, Inc