org.gradle.launcher.daemon.server.exec.LogToClient Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* 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.internal.logging.LoggingOutputInternal;
import org.gradle.internal.logging.events.OutputEvent;
import org.gradle.internal.logging.events.OutputEventListener;
import org.gradle.internal.logging.events.ProgressCompleteEvent;
import org.gradle.internal.logging.events.ProgressEvent;
import org.gradle.internal.logging.events.ProgressStartEvent;
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 java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.CountDownLatch;
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 Queue eventQueue = new ConcurrentLinkedQueue();
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 (dispatcher != null && (isMatchingBuildLogLevel(event) || isProgressEvent(event))) {
dispatcher.submit(event);
}
}
private boolean isProgressEvent(OutputEvent event) {
return event instanceof ProgressStartEvent || event instanceof ProgressEvent || event instanceof ProgressCompleteEvent;
}
private boolean isMatchingBuildLogLevel(OutputEvent event) {
return event.getLogLevel() != null && event.getLogLevel().compareTo(buildLogLevel) >= 0;
}
};
LOGGER.debug(DaemonMessages.ABOUT_TO_START_RELAYING_LOGS);
loggingOutput.addOutputEventListener(listener);
}
public void submit(OutputEvent event) {
eventQueue.add(event);
}
@Override
public void run() {
try {
while (!shouldStop) {
OutputEvent event = eventQueue.poll();
if (event == null) {
Thread.sleep(10);
} else {
dispatchAsync(event);
}
}
} catch (InterruptedException ex) {
// we must not use interrupt() because it would automatically
// close the connection (sending data from an interrupted thread
// automatically closes the connection)
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