sclass TailFile { File file; int interval; long l; new Flag stopped; java.util.Timer timer; ReliableSingleThread thread; volatile bool hasIdled, debug; *() {} *(final File *file, int *interval, fO onData) { thread = new ReliableSingleThread(r { long l2 = l(file); if (l2 < l) l = 0; // file shrunk (log rotated!) - start over from beginning of file if (l2 == l) hasIdled = true; else pcall { if (debug) print("New data in " + f2s(file) + ": " + n2(l2-l, "byte")); S text = null; for (S s : loadTextFilePart_iterator(file, l, l2)) pcallF(onData, s); l = l2; } }); } void start { timer = doEvery(interval, r { if (!stopped.isUp()) thread.trigger() }); } void stop { timer.cancel(); timer = null; if (stopped.raise()) thread.triggerAndWait(); // One last time baby } bool started() { ret timer != null; } void debugOn { if (debug) ret; debug = true; print("Watching file: " + f2s(file)); } }