Speed up spill writer. Echo failed test output to disk.

This commit is contained in:
Dawid Weiss 2020-01-08 10:55:07 +01:00
parent 14dd5a5e4d
commit 85d261339b
4 changed files with 179 additions and 47 deletions

View File

@ -1,14 +1,31 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.lucene.gradle;
import java.io.Closeable;
import java.io.IOException;
import java.io.PrintStream;
import java.io.UncheckedIOException;
import java.io.*;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.regex.Pattern;
import org.gradle.api.internal.tasks.testing.logging.FullExceptionFormatter;
import org.gradle.api.internal.tasks.testing.logging.TestExceptionFormatter;
import org.gradle.api.logging.Logger;
@ -32,22 +49,17 @@ public class ErrorReportingTestListener implements TestOutputListener, TestListe
private final TestExceptionFormatter formatter;
private final Map<TestKey, OutputHandler> outputHandlers = new ConcurrentHashMap<>();
private final Path spillDir;
private final Path outputsDir;
public ErrorReportingTestListener(TestLogging testLogging, Path spillDir) {
public ErrorReportingTestListener(TestLogging testLogging, Path spillDir, Path outputsDir) {
this.formatter = new FullExceptionFormatter(testLogging);
this.spillDir = spillDir;
this.outputsDir = outputsDir;
}
@Override
public void onOutput(TestDescriptor testDescriptor, TestOutputEvent outputEvent) {
TestDescriptor suite = testDescriptor.getParent();
// Check if this is output from the test suite itself (e.g. afterTest or beforeTest)
if (testDescriptor.isComposite()) {
suite = testDescriptor;
}
handlerFor(suite).write(outputEvent);
handlerFor(testDescriptor).write(outputEvent);
}
@Override
@ -62,51 +74,89 @@ public class ErrorReportingTestListener implements TestOutputListener, TestListe
@Override
public void afterSuite(final TestDescriptor suite, TestResult result) {
if (suite.getParent() == null || suite.getName().startsWith("Gradle")) {
return;
}
TestKey key = TestKey.of(suite);
try {
// if the test suite failed, report all captured output
if (Objects.equals(result.getResultType(), TestResult.ResultType.FAILURE)) {
reportFailure(suite, outputHandlers.get(key));
OutputHandler outputHandler = outputHandlers.get(key);
if (outputHandler != null) {
long length = outputHandler.length();
if (length > 1024 * 1024 * 10) {
LOGGER.warn(String.format(Locale.ROOT, "WARNING: Test %s wrote %,d bytes of output.",
suite.getName(),
length));
}
}
boolean echoOutput = Objects.equals(result.getResultType(), TestResult.ResultType.FAILURE);
boolean dumpOutput = echoOutput; // Force output dumping.
// If the test suite failed, report output.
if (dumpOutput || echoOutput) {
Files.createDirectories(outputsDir);
Path outputLog = outputsDir.resolve(getOutputLogName(suite));
// Save the output of a failing test to disk.
try (Writer w = Files.newBufferedWriter(outputLog, StandardCharsets.UTF_8)) {
if (outputHandler != null) {
outputHandler.copyTo(w);
}
}
if (echoOutput) {
synchronized (this) {
System.out.println("");
System.out.println(suite.getClassName() + " > test suite's output saved to " + outputLog + ", copied below:");
try (BufferedReader reader = Files.newBufferedReader(outputLog, StandardCharsets.UTF_8)) {
char[] buf = new char[1024];
int len;
while ((len = reader.read(buf)) >= 0) {
System.out.print(new String(buf, 0, len));
}
System.out.println();
}
}
}
}
} catch (IOException e) {
throw new UncheckedIOException("Error reading test suite output", e);
throw new UncheckedIOException(e);
} finally {
OutputHandler writer = outputHandlers.remove(key);
if (writer != null) {
OutputHandler handler = outputHandlers.remove(key);
if (handler != null) {
try {
writer.close();
handler.close();
} catch (IOException e) {
LOGGER.error("Failed to close test suite's event writer for: " + key, e);
LOGGER.error("Failed to close output handler for: " + key, e);
}
}
}
}
private void reportFailure(TestDescriptor suite, OutputHandler outputHandler) throws IOException {
if (outputHandler != null) {
synchronized (this) {
System.out.println("");
System.out.println(suite.getClassName() + " > test suite's output copied below:");
outputHandler.copyTo(System.out);
}
}
private static Pattern SANITIZE = Pattern.compile("[^a-zA-Z .\\-_0-9]+");
public static String getOutputLogName(TestDescriptor suite) {
return SANITIZE.matcher("OUTPUT-" + suite.getName() + ".txt").replaceAll("_");
}
@Override
public void afterTest(TestDescriptor testDescriptor, TestResult result) {
// Include test failure exception stacktrace(s) in test output log.
if (result.getResultType() == TestResult.ResultType.FAILURE) {
if (testDescriptor.getParent() != null) {
if (result.getExceptions().size() > 0) {
String message = formatter.format(testDescriptor, result.getExceptions());
handlerFor(testDescriptor.getParent()).write(message);
}
if (result.getExceptions().size() > 0) {
String message = formatter.format(testDescriptor, result.getExceptions());
handlerFor(testDescriptor).write(message);
}
}
}
private OutputHandler handlerFor(TestDescriptor suite) {
return outputHandlers.computeIfAbsent(TestKey.of(suite), (key) -> new OutputHandler());
private OutputHandler handlerFor(TestDescriptor descriptor) {
// Attach output of leaves (individual tests) to their parent.
if (!descriptor.isComposite()) {
descriptor = descriptor.getParent();
}
return outputHandlers.computeIfAbsent(TestKey.of(descriptor), (key) -> new OutputHandler());
}
public static class TestKey {
@ -183,6 +233,10 @@ public class ErrorReportingTestListener implements TestOutputListener, TestListe
write(sint, message);
}
public long length() throws IOException {
return buffer.length();
}
private void write(PrefixedWriter out, String message) {
try {
if (out != last) {
@ -195,10 +249,9 @@ public class ErrorReportingTestListener implements TestOutputListener, TestListe
}
}
public void copyTo(PrintStream out) throws IOException {
public void copyTo(Writer out) throws IOException {
flush();
buffer.copyTo(out);
out.println();
}
public void flush() throws IOException {

View File

@ -1,3 +1,19 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.lucene.gradle;
import java.io.IOException;

View File

@ -1,7 +1,22 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.lucene.gradle;
import java.io.IOException;
import java.io.PrintStream;
import java.io.Reader;
import java.io.StringWriter;
import java.io.Writer;
@ -27,6 +42,44 @@ public class SpillWriter extends Writer {
getSink(len).write(cbuf, off, len);
}
@Override
public void write(int c) throws IOException {
getSink(1).write(c);
}
@Override
public void write(char[] cbuf) throws IOException {
getSink(cbuf.length).write(cbuf);
}
@Override
public void write(String str) throws IOException {
getSink(str.length()).write(str);
}
@Override
public void write(String str, int off, int len) throws IOException {
getSink(len).write(str, off, len);
}
@Override
public Writer append(CharSequence csq) throws IOException {
getSink(csq.length()).append(csq);
return this;
}
@Override
public Writer append(CharSequence csq, int start, int end) throws IOException {
getSink(Math.max(0, end - start)).append(csq, start, end);
return this;
}
@Override
public Writer append(char c) throws IOException {
getSink(1).append(c);
return this;
}
private Writer getSink(int expectedWriteChars) throws IOException {
if (spill == null) {
if (buffer.getBuffer().length() + expectedWriteChars <= MAX_BUFFERED) {
@ -56,18 +109,23 @@ public class SpillWriter extends Writer {
}
}
public void copyTo(PrintStream ps) throws IOException {
public void copyTo(Writer writer) throws IOException {
if (spill != null) {
flush();
char [] buf = new char [MAX_BUFFERED];
try (Reader reader = Files.newBufferedReader(spillPath, StandardCharsets.UTF_8)) {
int len;
while ((len = reader.read(buf)) >= 0) {
ps.print(new String(buf, 0, len));
}
reader.transferTo(writer);
}
} else {
ps.append(buffer.getBuffer());
writer.append(buffer.getBuffer());
}
}
public long length() throws IOException {
flush();
if (spill != null) {
return Files.size(spillPath);
} else {
return buffer.getBuffer().length();
}
}
}

View File

@ -98,8 +98,13 @@ allprojects {
}
// Set up custom test output handler.
def testOutputsDir = file("${reports.junitXml.destination}/outputs")
doFirst {
project.delete testOutputsDir
}
def spillDir = getTemporaryDir().toPath()
def listener = new ErrorReportingTestListener(test.testLogging, spillDir)
def listener = new ErrorReportingTestListener(test.testLogging, spillDir, testOutputsDir.toPath())
addTestOutputListener(listener)
addTestListener(listener)
}