/* * 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.karaf.log.command; import java.io.IOException; import java.io.PrintStream; import java.util.concurrent.BlockingQueue; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.LinkedBlockingQueue; import org.apache.karaf.log.core.LogService; import org.apache.karaf.shell.api.action.Command; import org.apache.karaf.shell.api.action.lifecycle.Reference; import org.apache.karaf.shell.api.action.lifecycle.Service; import org.apache.karaf.shell.api.console.Session; import org.ops4j.pax.logging.spi.PaxAppender; import org.ops4j.pax.logging.spi.PaxLoggingEvent; @Command(scope = "log", name = "tail", description = "Continuously display log entries. Use ctrl-c to quit this command") @Service public class LogTail extends DisplayLog { @Reference Session session; @Reference LogService logService; private ExecutorService executorService = Executors.newFixedThreadPool(2); @Override public Object execute() throws Exception { PrintEventThread printThread = new PrintEventThread(); ReadKeyBoardThread readKeyboardThread = new ReadKeyBoardThread(Thread.currentThread()); executorService.execute(printThread); executorService.execute(readKeyboardThread); while (!Thread.currentThread().isInterrupted()) { try { Thread.sleep(200); } catch (java.lang.InterruptedException e) { break; } } printThread.abort(); readKeyboardThread.abort(); executorService.shutdownNow(); return null; } class ReadKeyBoardThread implements Runnable { private Thread sessionThread; boolean readKeyboard = true; public ReadKeyBoardThread(Thread thread) { this.sessionThread = thread; } public void abort() { readKeyboard = false; } public void run() { while (readKeyboard) { try { int c = session.getKeyboard().read(); if (c < 0) { sessionThread.interrupt(); break; } } catch (IOException e) { break; } } } } class PrintEventThread implements Runnable { PrintStream out = System.out; boolean doDisplay = true; public void run() { int minLevel = getMinLevel(level); Iterable<PaxLoggingEvent> le = logService.getEvents(entries == 0 ? Integer.MAX_VALUE : entries); for (PaxLoggingEvent event : le) { printEvent(out, event, minLevel); } out.flush(); // Tail final BlockingQueue<PaxLoggingEvent> queue = new LinkedBlockingQueue<PaxLoggingEvent>(); PaxAppender appender = new PaxAppender() { public void doAppend(PaxLoggingEvent event) { queue.add(event); } }; try { logService.addAppender(appender); while (doDisplay) { printEvent(out, queue.take(), minLevel); if (queue.isEmpty()) { out.flush(); } } } catch (InterruptedException e) { // Ignore } finally { logService.removeAppender(appender); } out.println(); } public void abort() { doDisplay = false; } } }