48
48
import java .util .logging .Level ;
49
49
import java .util .logging .Logger ;
50
50
import org .apache .commons .io .input .NullReader ;
51
+ import org .apache .commons .io .output .CountingOutputStream ;
51
52
import org .jenkinsci .plugins .workflow .flow .FlowExecutionOwner ;
52
53
import org .jenkinsci .plugins .workflow .graph .FlowNode ;
53
54
import org .kohsuke .accmod .Restricted ;
58
59
* Simple implementation of log storage in a single file that maintains a side file with an index indicating where node transitions occur.
59
60
* Each line in the index file is a byte offset, optionally followed by a space and then a node ID.
60
61
*/
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
+ */
61
65
@ Restricted (Beta .class )
62
66
public final class FileLogStorage implements LogStorage {
63
67
@@ -73,6 +77,10 @@ public static synchronized LogStorage forFile(File log) {
73
77
private final File index ;
74
78
@ SuppressFBWarnings (value = "IS2_INCONSISTENT_SYNC" , justification = "actually it is always accessed within the monitor" )
75
79
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 ;
76
84
@ SuppressFBWarnings (value = "IS2_INCONSISTENT_SYNC" , justification = "we only care about synchronizing writes" )
77
85
private OutputStream bos ;
78
86
private Writer indexOs ;
@@ -86,7 +94,9 @@ private FileLogStorage(File log) {
86
94
private synchronized void open () throws IOException {
87
95
if (os == null ) {
88
96
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 ));
90
100
if (index .isFile ()) {
91
101
try (BufferedReader r = Files .newBufferedReader (index .toPath (), StandardCharsets .UTF_8 )) {
92
102
// 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 {
126
136
assert Thread .holdsLock (this );
127
137
if (!Objects .equals (id , lastId )) {
128
138
bos .flush ();
129
- long pos = os . getChannel (). position ();
139
+ long pos = osStartPosition + cos . getByteCount ();
130
140
if (id == null ) {
131
141
indexOs .write (pos + "\n " );
132
142
} else {
0 commit comments