org.gradle.launcher.daemon.server.exec.LogToClient Maven / Gradle / Ivy
/*
* Copyright 2011 the original author or authors.
*
* 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 org.gradle.launcher.daemon.server.exec;
import org.gradle.api.logging.LogLevel;
import org.gradle.api.logging.Logger;
import org.gradle.api.logging.Logging;
import org.gradle.launcher.daemon.diagnostics.DaemonDiagnostics;
import org.gradle.launcher.daemon.logging.DaemonMessages;
import org.gradle.launcher.daemon.protocol.Build;
import org.gradle.launcher.daemon.server.api.DaemonCommandExecution;
import org.gradle.launcher.daemon.server.api.DaemonConnection;
import org.gradle.internal.logging.LoggingOutputInternal;
import org.gradle.internal.logging.events.OutputEvent;
import org.gradle.internal.logging.events.OutputEventListener;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.TimeUnit;
public class LogToClient extends BuildCommandOnly {
public static final String DISABLE_OUTPUT = "org.gradle.daemon.disable-output";
private static final Logger LOGGER = Logging.getLogger(LogToClient.class);
private final LoggingOutputInternal loggingOutput;
private final DaemonDiagnostics diagnostics;
private volatile AsynchronousLogDispatcher dispatcher;
public LogToClient(LoggingOutputInternal loggingOutput, DaemonDiagnostics diagnostics) {
this.loggingOutput = loggingOutput;
this.diagnostics = diagnostics;
}
protected void doBuild(final DaemonCommandExecution execution, Build build) {
if (Boolean.getBoolean(DISABLE_OUTPUT)) {
execution.proceed();
return;
}
dispatcher = new AsynchronousLogDispatcher(execution.getConnection(), build.getParameters().getLogLevel());
LOGGER.info("{}{}). The daemon log file: {}", DaemonMessages.STARTED_RELAYING_LOGS, diagnostics.getPid(), diagnostics.getDaemonLog());
dispatcher.start();
try {
execution.proceed();
} finally {
dispatcher.waitForCompletion();
}
}
private class AsynchronousLogDispatcher extends Thread {
private final CountDownLatch completionLock = new CountDownLatch(1);
private final BlockingQueue eventQueue = new LinkedBlockingDeque();
private final DaemonConnection connection;
private final OutputEventListener listener;
private volatile boolean shouldStop;
private boolean unableToSend;
private AsynchronousLogDispatcher(DaemonConnection conn, final LogLevel buildLogLevel) {
super("Asynchronous log dispatcher for " + conn);
this.connection = conn;
this.listener = new OutputEventListener() {
public void onOutput(OutputEvent event) {
if (event.getLogLevel() != null && event.getLogLevel().compareTo(buildLogLevel) >= 0) {
dispatcher.submit(event);
}
}
};
LOGGER.debug(DaemonMessages.ABOUT_TO_START_RELAYING_LOGS);
loggingOutput.addOutputEventListener(listener);
}
public void submit(OutputEvent event) {
eventQueue.add(event);
}
@Override
public void run() {
OutputEvent event;
try {
while (!shouldStop) {
// we must not use interrupt() because it would automatically
// close the connection (sending data from an interrupted thread
// automatically closes the connection)
event = eventQueue.poll(10, TimeUnit.MILLISECONDS);
if (event != null) {
dispatchAsync(event);
}
}
} catch (InterruptedException ex) {
shouldStop = true;
}
sendRemainingEvents();
completionLock.countDown();
}
private void sendRemainingEvents() {
OutputEvent event;
while ((event = eventQueue.poll()) != null) {
dispatchAsync(event);
}
}
private void dispatchAsync(OutputEvent event) {
if (unableToSend) {
return;
}
try {
connection.logEvent(event);
} catch (Exception ex) {
shouldStop = true;
unableToSend = true;
//Ignore. It means the client has disconnected so no point sending him any log output.
//we should be checking if client still listens elsewhere anyway.
}
}
public void waitForCompletion() {
loggingOutput.removeOutputEventListener(listener);
shouldStop = true;
try {
completionLock.await();
} catch (InterruptedException e) {
// the caller has been interrupted
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy