blob: f133ce9f39f2d221dfe2e1402a9dfaddf415b0e5 [file] [log] [blame]
// Copyright 2018 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.server;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.server.CommandProtos.CancelRequest;
import com.google.devtools.build.lib.util.ThreadUtils;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.atomic.AtomicLong;
import java.util.logging.Logger;
import javax.annotation.concurrent.GuardedBy;
/** Helper class for commands that are currently running on the server. */
class CommandManager {
private static final Logger logger = Logger.getLogger(CommandManager.class.getName());
@GuardedBy("runningCommandsMap")
private final Map<String, RunningCommand> runningCommandsMap = new HashMap<>();
private final AtomicLong interruptCounter = new AtomicLong(0);
private final boolean doIdleServerTasks;
private IdleServerTasks idleServerTasks;
CommandManager(boolean doIdleServerTasks) {
this.doIdleServerTasks = doIdleServerTasks;
idle();
}
void interruptInflightCommands() {
synchronized (runningCommandsMap) {
for (RunningCommand command : runningCommandsMap.values()) {
command.thread.interrupt();
}
startSlowInterruptWatcher(ImmutableSet.copyOf(runningCommandsMap.keySet()));
}
}
void doCancel(CancelRequest request) {
try (RunningCommand cancelCommand = create()) {
synchronized (runningCommandsMap) {
RunningCommand pendingCommand = runningCommandsMap.get(request.getCommandId());
if (pendingCommand != null) {
logger.info(
String.format(
"Interrupting command %s on thread %s",
request.getCommandId(), pendingCommand.thread.getName()));
pendingCommand.thread.interrupt();
startSlowInterruptWatcher(ImmutableSet.of(request.getCommandId()));
} else {
logger.info("Cannot find command " + request.getCommandId() + " to interrupt");
}
}
}
}
boolean isEmpty() {
synchronized (runningCommandsMap) {
return runningCommandsMap.isEmpty();
}
}
void waitForChange() throws InterruptedException {
synchronized (runningCommandsMap) {
runningCommandsMap.wait();
}
}
void waitForChange(long timeout) throws InterruptedException {
synchronized (runningCommandsMap) {
runningCommandsMap.wait(timeout);
}
}
RunningCommand create() {
RunningCommand command = new RunningCommand();
synchronized (runningCommandsMap) {
if (runningCommandsMap.isEmpty()) {
busy();
}
runningCommandsMap.put(command.id, command);
runningCommandsMap.notify();
}
logger.info(
String.format("Starting command %s on thread %s", command.id, command.thread.getName()));
return command;
}
private void idle() {
Preconditions.checkState(idleServerTasks == null);
if (doIdleServerTasks) {
idleServerTasks = new IdleServerTasks();
idleServerTasks.idle();
}
}
private void busy() {
if (doIdleServerTasks) {
Preconditions.checkState(idleServerTasks != null);
idleServerTasks.busy();
idleServerTasks = null;
}
}
private void startSlowInterruptWatcher(final ImmutableSet<String> commandIds) {
if (commandIds.isEmpty()) {
return;
}
Runnable interruptWatcher =
() -> {
try {
Thread.sleep(10 * 1000);
boolean ok;
synchronized (runningCommandsMap) {
ok = Collections.disjoint(commandIds, runningCommandsMap.keySet());
}
if (!ok) {
// At least one command was not interrupted. Interrupt took too long.
ThreadUtils.warnAboutSlowInterrupt();
}
} catch (InterruptedException e) {
// Ignore.
}
};
Thread interruptWatcherThread =
new Thread(interruptWatcher, "interrupt-watcher-" + interruptCounter.incrementAndGet());
interruptWatcherThread.setDaemon(true);
interruptWatcherThread.start();
}
class RunningCommand implements AutoCloseable {
private final Thread thread;
private final String id;
private RunningCommand() {
thread = Thread.currentThread();
id = UUID.randomUUID().toString();
}
@Override
public void close() {
synchronized (runningCommandsMap) {
runningCommandsMap.remove(id);
if (runningCommandsMap.isEmpty()) {
idle();
}
runningCommandsMap.notify();
}
logger.info(String.format("Finished command %s on thread %s", id, thread.getName()));
}
String getId() {
return id;
}
}
}