Реализация Java IO для unix/linux "tail -f"
Мне интересно, какие методы и/или библиотеки использовать для реализации функциональности команды linux "tail -f". Я по существу ищу возможность добавления/замены для java.io.FileReader
. Код клиента может выглядеть примерно так:
TailFileReader lft = new TailFileReader("application.log");
BufferedReader br = new BufferedReader(lft);
String line;
try {
while (true) {
line= br.readLine();
// do something interesting with line
}
} catch (IOException e) {
// barf
}
Пропущенный фрагмент является разумной реализацией TailFileReader
. Он должен иметь возможность читать части файла, которые существуют до открытия файла, а также строки, которые добавляются.
Ответы
Ответ 1
Возможность продолжить чтение файла и подождать до тех пор, пока у файла не появится больше обновлений для вас, это не так сложно выполнить в коде. Вот несколько псевдокодов:
BufferedReader br = new BufferedReader(...);
String line;
while (keepReading) {
line = reader.readLine();
if (line == null) {
//wait until there is more of the file for us to read
Thread.sleep(1000);
}
else {
//do something interesting with the line
}
}
Я бы предположил, что вы захотите включить этот тип функций в свой собственный поток, чтобы вы могли спать и не влияли на другие области вашего приложения. Вы хотите показать keepReading
в сеттере, чтобы ваш основной класс/другие части приложения могли безопасно закрыть поток без каких-либо других головных болей, просто позвонив stopReading()
или что-то подобное.
Ответ 2
Взгляните на реализацию Apache Commons класса Tailer. Кажется, что он также обрабатывает вращение журнала.
Ответ 3
Отметьте JLogTailer, который выполняет эту логику.
Основной смысл в коде:
public void run() {
try {
while (_running) {
Thread.sleep(_updateInterval);
long len = _file.length();
if (len < _filePointer) {
// Log must have been jibbled or deleted.
this.appendMessage("Log file was reset. Restarting logging from start of file.");
_filePointer = len;
}
else if (len > _filePointer) {
// File must have had something added to it!
RandomAccessFile raf = new RandomAccessFile(_file, "r");
raf.seek(_filePointer);
String line = null;
while ((line = raf.readLine()) != null) {
this.appendLine(line);
}
_filePointer = raf.getFilePointer();
raf.close();
}
}
}
catch (Exception e) {
this.appendMessage("Fatal error reading log file, log tailing has stopped.");
}
// dispose();
}
Ответ 4
Я построил короткую реализацию "tail -f" в Scala некоторое время назад: tailf. Он также заботится о вращении файла, и вы можете определить свою собственную логику, что делать, когда он достигнет EOF или обнаружил, что файл был переименован.
Вы можете взглянуть и перенести его на Java, поскольку на самом деле там нет ничего сложного. Несколько заметок: основной файл Tail.scala и в основном он определяет FollowingInputStream
, который заботится об EOF/rename и follow
, который переносит FollowingInputStream
в неограниченное перечисление в SequenceInputStream
. Итак, как только FollowingInputStream
заканчивается, SequenceInputStream
запрашивает следующий элемент из Enumeration
и создается FollowingInputStream
.
Ответ 5
Я недавно наткнулся на rxjava-file. Это расширение RxJava. В отличие от других решений, это использует Java NIO.
import rx.Observable;
import rx.functions.Action1;
import com.github.davidmoten.rx.FileObservable;
// ... class definition omitted
public void tailLogFile() throws InterruptedException {
Observable<String> tailer = FileObservable.tailer()
.file("application.log") // absolute path
.tailText();
tailer.subscribe(
new Action1<String>() {
@Override
public void call(String line) {
System.out.println("you got line: " + line);
}
},
new Action1<Throwable>() {
@Override
public void call(Throwable e) {
System.out.println("you got error: " + e);
e.printStackTrace();
}
}
);
// this solution operates threaded, so something
// is required that prevents premature termination
Thread.sleep(120000);
}
Ответ 6
Вот короткая история, которую вы могли бы использовать в качестве указателя:
Я закодировал TailingInputStream на работе по той же причине. Он в основном использует файл и обновляет его содержимое по требованию и проверяет его на внутренний буфер, если он значительно изменился (4KB-память штампа IIRC), а затем сделал то, что делает tail -f. Немного взломанный, да, но он отлично работает и не вмешивается в Threads или что-то вроде этого - он полностью совместим с 1.4.2 по крайней мере.
Тем не менее, было намного проще сделать, чем ReverseInputStream, который перешел от конца файла к началу и не умер, если файл обновлялся на лету...
Ответ 7
Если ваш код когда-либо будет запускаться в Unix-системах, вы можете уйти с помощью простого обхода и прямого вызова tail -f
.
В качестве более привлекательной альтернативы вы можете взглянуть на реализацию хвоста GNU и порта, который связан с Java. (Я не уверен, что это еще не сделает ваш код производным.)
Ответ 8
Просто столкнулся с той же проблемой - здесь была найдена "простейшая" реализация: Java Tail.
* Отличный материал * - готов к производству;)
Я надеюсь, что код-цитата не откажет в лицензии.
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
/**
* Java implementation of the Unix tail command
*
* @param args[0] File name
* @param args[1] Update time (seconds). Optional. Default value is 1 second
*
* @author Luigi Viggiano (original author) http://it.newinstance.it/2005/11/19/listening-changes-on-a-text-file-unix-tail-implementation-with-java/
* @author Alessandro Melandri (modified by)
* */
public class Tail {
static long sleepTime = 1000;
public static void main(String[] args) throws IOException {
if (args.length > 0){
if (args.length > 1)
sleepTime = Long.parseLong(args[1]) * 1000;
BufferedReader input = new BufferedReader(new FileReader(args[0]));
String currentLine = null;
while (true) {
if ((currentLine = input.readLine()) != null) {
System.out.println(currentLine);
continue;
}
try {
Thread.sleep(sleepTime);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
break;
}
}
input.close();
} else {
System.out.println("Missing parameter!\nUsage: java JavaTail fileName [updateTime (Seconds. default to 1 second)]");
}
}
}
Ответ 9
Я нашел эту приятную реализацию хвоста.
Автор: amelandri
Суе из: https://gist.github.com/amelandri/1376896
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
/**
* Java implementation of the Unix tail command
*
* @param args[0] File name
* @param args[1] Update time (seconds). Optional. Default value is 1 second
*
* @author Luigi Viggiano (original author) http://it.newinstance.it/2005/11/19/listening-changes-on-a-text-file-unix-tail-implementation-with-java/
* @author Alessandro Melandri (modified by)
* */
public class Tail {
static long sleepTime = 1000;
public static void main(String[] args) throws IOException {
if (args.length > 0){
if (args.length > 1)
sleepTime = Long.parseLong(args[1]) * 1000;
BufferedReader input = new BufferedReader(new FileReader(args[0]));
String currentLine = null;
while (true) {
if ((currentLine = input.readLine()) != null) {
System.out.println(currentLine);
continue;
}
try {
Thread.sleep(sleepTime);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
break;
}
}
input.close();
} else {
System.out.println("Missing parameter!\nUsage: java JavaTail fileName [updateTime (Seconds. default to 1 second)]");
}
}
}