blob: 1f51559d87f377bdd3b286e1d901ac02c54e1955 [file] [log] [blame]
// Copyright 2019 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.remote.util;
import build.bazel.remote.execution.v2.ExecutionGrpc;
import com.google.common.base.Stopwatch;
import com.google.devtools.build.lib.concurrent.ThreadSafety;
import io.grpc.CallOptions;
import io.grpc.Channel;
import io.grpc.ClientCall;
import io.grpc.ClientInterceptor;
import io.grpc.Context;
import io.grpc.ForwardingClientCall;
import io.grpc.ForwardingClientCallListener;
import io.grpc.Metadata;
import io.grpc.MethodDescriptor;
import io.grpc.Status;
import java.time.Duration;
/** Reentrant wall clock stopwatch and grpc interceptor for network waits. */
@ThreadSafety.ThreadSafe
public class NetworkTime {
public static final Context.Key<NetworkTime> CONTEXT_KEY = Context.key("remote-network-time");
private final Stopwatch wallTime = Stopwatch.createUnstarted();
private int outstanding = 0;
private synchronized void start() {
if (!wallTime.isRunning()) {
wallTime.start();
}
outstanding++;
}
private synchronized void stop() {
if (--outstanding == 0) {
wallTime.stop();
}
}
public Duration getDuration() {
return wallTime.elapsed();
}
/** The ClientInterceptor used to track network time. */
public static class Interceptor implements ClientInterceptor {
@Override
public <ReqT, RespT> ClientCall<ReqT, RespT> interceptCall(
MethodDescriptor<ReqT, RespT> method, CallOptions callOptions, Channel next) {
ClientCall<ReqT, RespT> call = next.newCall(method, callOptions);
// prevent accounting for execution wait time
if (method != ExecutionGrpc.getExecuteMethod()
&& method != ExecutionGrpc.getWaitExecutionMethod()) {
NetworkTime networkTime = CONTEXT_KEY.get();
if (networkTime != null) {
call = new NetworkTimeCall<>(call, networkTime);
}
}
return call;
}
}
private static class NetworkTimeCall<ReqT, RespT>
extends ForwardingClientCall.SimpleForwardingClientCall<ReqT, RespT> {
private final NetworkTime networkTime;
private boolean firstMessage = true;
protected NetworkTimeCall(ClientCall<ReqT, RespT> delegate, NetworkTime networkTime) {
super(delegate);
this.networkTime = networkTime;
}
@Override
public void start(Listener<RespT> responseListener, Metadata headers) {
super.start(
new ForwardingClientCallListener.SimpleForwardingClientCallListener<RespT>(
responseListener) {
@Override
public void onClose(Status status, Metadata trailers) {
networkTime.stop();
super.onClose(status, trailers);
}
},
headers);
}
@Override
public void sendMessage(ReqT message) {
if (firstMessage) {
networkTime.start();
firstMessage = false;
}
super.sendMessage(message);
}
}
}