/*
*
* Copyright 2014 http://Bither.net
*
* Licensed 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 net.bither.logging;
import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.Appender;
import ch.qos.logback.core.AppenderBase;
import com.google.common.collect.Lists;
import com.google.common.collect.Queues;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import java.util.List;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ThreadFactory;
/**
* <p>Appender to provide the following to logging framework:</p>
* <ul>
* <li>Asynchronous logging</li>
* </ul>
*
* @since 0.0.1
*
*/
public class AsyncAppender extends AppenderBase<ILoggingEvent> implements Runnable {
private static final int BATCH_SIZE = 1000;
public static Appender<ILoggingEvent> wrap(Appender<ILoggingEvent> delegate) {
final AsyncAppender appender = new AsyncAppender(delegate);
appender.start();
return appender;
}
private static final ThreadFactory THREAD_FACTORY =
new ThreadFactoryBuilder().setNameFormat("async-log-appender-%d")
.setDaemon(true)
.build();
private final Appender<ILoggingEvent> delegate;
private final BlockingQueue<ILoggingEvent> queue;
private final List<ILoggingEvent> batch;
private final Thread dispatcher;
private volatile boolean running;
private AsyncAppender(Appender<ILoggingEvent> delegate) {
this.delegate = delegate;
this.queue = Queues.newLinkedBlockingQueue();
this.batch = Lists.newArrayListWithCapacity(BATCH_SIZE);
this.dispatcher = THREAD_FACTORY.newThread(this);
setContext(delegate.getContext());
}
@Override
protected void append(ILoggingEvent eventObject) {
eventObject.prepareForDeferredProcessing();
queue.add(eventObject);
}
@Override
public void start() {
super.start();
this.running = true;
dispatcher.start();
}
@Override
public void stop() {
this.running = false;
super.stop();
}
@Override
public void run() {
while (running) {
try {
batch.add(queue.take());
queue.drainTo(batch, BATCH_SIZE - 1);
for (ILoggingEvent event : batch) {
delegate.doAppend(event);
}
batch.clear();
} catch (InterruptedException ignored) {
Thread.currentThread().interrupt();
}
}
}
}