org.gradle.process.internal.streams.ExecOutputHandleRunner Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-test-kit Show documentation
Show all versions of gradle-test-kit Show documentation
Gradle 6.2.1 API redistribution.
/*
* Copyright 2010 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.process.internal.streams;
import org.gradle.api.logging.Logger;
import org.gradle.api.logging.Logging;
import org.gradle.internal.concurrent.CompositeStoppable;
import org.gradle.internal.operations.BuildOperationRef;
import org.gradle.internal.operations.CurrentBuildOperationRef;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.concurrent.CountDownLatch;
public class ExecOutputHandleRunner implements Runnable {
private final static Logger LOGGER = Logging.getLogger(ExecOutputHandleRunner.class);
private final String displayName;
private final InputStream inputStream;
private final OutputStream outputStream;
private final int bufferSize;
private final CountDownLatch completed;
private volatile boolean closed;
private volatile BuildOperationRef associatedBuildOperation;
public ExecOutputHandleRunner(String displayName, InputStream inputStream, OutputStream outputStream, CountDownLatch completed) {
this(displayName, inputStream, outputStream, 8192, completed);
}
ExecOutputHandleRunner(String displayName, InputStream inputStream, OutputStream outputStream, int bufferSize, CountDownLatch completed) {
this.displayName = displayName;
this.inputStream = inputStream;
this.outputStream = outputStream;
this.bufferSize = bufferSize;
this.completed = completed;
}
public void associateBuildOperation(BuildOperationRef startupRef) {
this.associatedBuildOperation = startupRef;
}
public void clearAssociatedBuildOperation() {
this.associatedBuildOperation = null;
}
@Override
public void run() {
try {
forwardContent();
} finally {
completed.countDown();
}
}
private void forwardContent() {
try {
byte[] buffer = new byte[bufferSize];
while (!closed) {
int nread = inputStream.read(buffer);
if (nread < 0) {
break;
}
BuildOperationRef startupRef = this.associatedBuildOperation;
if (startupRef != null) {
CurrentBuildOperationRef.instance().with(startupRef, () -> {
writeBuffer(buffer, nread);
return null;
});
} else {
writeBuffer(buffer, nread);
}
}
CompositeStoppable.stoppable(inputStream, outputStream).stop();
} catch (Throwable t) {
if (!closed && !wasInterrupted(t)) {
LOGGER.error(String.format("Could not %s.", displayName), t);
}
}
}
private void writeBuffer(byte[] buffer, int nread) throws IOException {
outputStream.write(buffer, 0, nread);
outputStream.flush();
}
/**
* This can happen e.g. on IBM JDK when a remote process was terminated. Instead of
* returning -1 on the next read() call, it will interrupt the current read call.
*/
private boolean wasInterrupted(Throwable t) {
return t instanceof IOException && "Interrupted system call".equals(t.getMessage());
}
public void closeInput() throws IOException {
disconnect();
inputStream.close();
}
@Override
public String toString() {
return displayName;
}
public void disconnect() {
closed = true;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy