Skip to content

Commit d7c4973

Browse files
authored
Merge pull request jenkinsci#296 from dwnusbaum/interruption-robustness
[JENKINS-56446] Do not permanently close the log stream in FileLogStorage if an interrupted thread attempts to write to it
2 parents 5766370 + b763772 commit d7c4973

File tree

3 files changed

+27
-3
lines changed

3 files changed

+27
-3
lines changed

src/main/java/org/jenkinsci/plugins/workflow/log/FileLogStorage.java

+12-2
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@
4848
import java.util.logging.Level;
4949
import java.util.logging.Logger;
5050
import org.apache.commons.io.input.NullReader;
51+
import org.apache.commons.io.output.CountingOutputStream;
5152
import org.jenkinsci.plugins.workflow.flow.FlowExecutionOwner;
5253
import org.jenkinsci.plugins.workflow.graph.FlowNode;
5354
import org.kohsuke.accmod.Restricted;
@@ -58,6 +59,9 @@
5859
* Simple implementation of log storage in a single file that maintains a side file with an index indicating where node transitions occur.
5960
* Each line in the index file is a byte offset, optionally followed by a space and then a node ID.
6061
*/
62+
/* Note: Avoid FileChannel methods in this class, as they close the channel and its parent stream if the thread is
63+
interrupted, which is problematic given that we do not control the threads which write to the log file.
64+
*/
6165
@Restricted(Beta.class)
6266
public final class FileLogStorage implements LogStorage {
6367

@@ -73,6 +77,10 @@ public static synchronized LogStorage forFile(File log) {
7377
private final File index;
7478
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "actually it is always accessed within the monitor")
7579
private FileOutputStream os;
80+
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "actually it is always accessed within the monitor")
81+
private long osStartPosition;
82+
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "actually it is always accessed within the monitor")
83+
private CountingOutputStream cos;
7684
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "we only care about synchronizing writes")
7785
private OutputStream bos;
7886
private Writer indexOs;
@@ -86,7 +94,9 @@ private FileLogStorage(File log) {
8694
private synchronized void open() throws IOException {
8795
if (os == null) {
8896
os = new FileOutputStream(log, true);
89-
bos = new GCFlushedOutputStream(new DelayBufferedOutputStream(os));
97+
osStartPosition = log.length();
98+
cos = new CountingOutputStream(os);
99+
bos = new GCFlushedOutputStream(new DelayBufferedOutputStream(cos));
90100
if (index.isFile()) {
91101
try (BufferedReader r = Files.newBufferedReader(index.toPath(), StandardCharsets.UTF_8)) {
92102
// TODO would be faster to scan the file backwards for the penultimate \n, then convert the byte sequence from there to EOF to UTF-8 and set lastId accordingly
@@ -126,7 +136,7 @@ private void checkId(String id) throws IOException {
126136
assert Thread.holdsLock(this);
127137
if (!Objects.equals(id, lastId)) {
128138
bos.flush();
129-
long pos = os.getChannel().position();
139+
long pos = osStartPosition + cos.getByteCount();
130140
if (id == null) {
131141
indexOs.write(pos + "\n");
132142
} else {

src/test/java/org/jenkinsci/plugins/workflow/log/FileLogStorageTest.java

+14
Original file line numberDiff line numberDiff line change
@@ -55,4 +55,18 @@ public class FileLogStorageTest extends LogStorageTestBase {
5555
assertOverallLog(0, lines("stuff"), true);
5656
}
5757

58+
@Test public void interruptionDoesNotCloseStream() throws Exception {
59+
LogStorage ls = createStorage();
60+
TaskListener overall = ls.overallListener();
61+
overall.getLogger().println("overall 1");
62+
Thread.currentThread().interrupt();
63+
TaskListener stepLog = ls.nodeListener(new MockNode("1"));
64+
stepLog.getLogger().println("step 1");
65+
assertTrue(Thread.interrupted());
66+
close(stepLog);
67+
overall.getLogger().println("overall 2");
68+
close(overall);
69+
assertOverallLog(0, lines("overall 1", "<span class=\"pipeline-node-1\">step 1", "</span>overall 2"), true);
70+
}
71+
5872
}

src/test/java/org/jenkinsci/plugins/workflow/log/LogStorageTestBase.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -350,7 +350,7 @@ protected String lines(CharSequence... lines) {
350350
return String.join(System.lineSeparator(), lines) + System.lineSeparator();
351351
}
352352

353-
private static class MockNode extends FlowNode {
353+
protected static class MockNode extends FlowNode {
354354
MockNode(String id) {super(new MockFlowExecution(), id);}
355355
@Override protected String getTypeDisplayName() {return null;}
356356
}

0 commit comments

Comments
 (0)