com.internetitem.logback.elasticsearch.AbstractElasticsearchPublisher Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of logback-elasticsearch-appender Show documentation
Show all versions of logback-elasticsearch-appender Show documentation
Send log events directly from Logback to Elasticsearch
package com.internetitem.logback.elasticsearch;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import javax.xml.bind.DatatypeConverter;
import ch.qos.logback.core.Context;
import com.fasterxml.jackson.core.JsonFactory;
import com.fasterxml.jackson.core.JsonGenerator;
import com.internetitem.logback.elasticsearch.config.ElasticsearchProperties;
import com.internetitem.logback.elasticsearch.config.HttpRequestHeaders;
import com.internetitem.logback.elasticsearch.config.Property;
import com.internetitem.logback.elasticsearch.config.Settings;
import com.internetitem.logback.elasticsearch.util.AbstractPropertyAndEncoder;
import com.internetitem.logback.elasticsearch.util.ErrorReporter;
import com.internetitem.logback.elasticsearch.writer.ElasticsearchWriter;
import com.internetitem.logback.elasticsearch.writer.LoggerWriter;
import com.internetitem.logback.elasticsearch.writer.StdErrWriter;
public abstract class AbstractElasticsearchPublisher implements Runnable {
private static final AtomicInteger THREAD_COUNTER = new AtomicInteger(1);
public static final String THREAD_NAME_PREFIX = "es-writer-";
private volatile List events;
private ElasticsearchOutputAggregator outputAggregator;
private List> propertyList;
private AbstractPropertyAndEncoder indexPattern;
private JsonFactory jf;
private JsonGenerator jsonGenerator;
private ErrorReporter errorReporter;
protected Settings settings;
private final Object lock;
private volatile boolean working;
public AbstractElasticsearchPublisher(Context context, ErrorReporter errorReporter, Settings settings, ElasticsearchProperties properties, HttpRequestHeaders headers) throws IOException {
this.errorReporter = errorReporter;
this.events = new ArrayList();
this.lock = new Object();
this.settings = settings;
this.outputAggregator = configureOutputAggregator(settings, errorReporter, headers);
this.jf = new JsonFactory();
this.jf.setRootValueSeparator(null);
this.jsonGenerator = jf.createGenerator(outputAggregator);
this.indexPattern = buildPropertyAndEncoder(context, new Property("", settings.getIndex(), false));
this.propertyList = generatePropertyList(context, properties);
}
private static ElasticsearchOutputAggregator configureOutputAggregator(Settings settings, ErrorReporter errorReporter, HttpRequestHeaders httpRequestHeaders) {
ElasticsearchOutputAggregator spigot = new ElasticsearchOutputAggregator(settings, errorReporter);
if (settings.isLogsToStderr()) {
spigot.addWriter(new StdErrWriter());
}
if (settings.getLoggerName() != null) {
spigot.addWriter(new LoggerWriter(settings.getLoggerName()));
}
if (settings.getUrl() != null) {
spigot.addWriter(new ElasticsearchWriter(errorReporter, settings, httpRequestHeaders));
}
return spigot;
}
private List> generatePropertyList(Context context, ElasticsearchProperties properties) {
List> list = new ArrayList>();
if (properties != null) {
for (Property property : properties.getProperties()) {
list.add(buildPropertyAndEncoder(context, property));
}
}
return list;
}
protected abstract AbstractPropertyAndEncoder buildPropertyAndEncoder(Context context, Property property);
public void addEvent(T event) {
if (!outputAggregator.hasOutputs()) {
return;
}
synchronized (lock) {
events.add(event);
if (!working) {
working = true;
Thread thread = new Thread(this, THREAD_NAME_PREFIX + THREAD_COUNTER.getAndIncrement());
thread.start();
}
}
}
public void run() {
int currentTry = 1;
int maxRetries = settings.getMaxRetries();
while (true) {
try {
Thread.sleep(settings.getSleepTime());
List eventsCopy = null;
synchronized (lock) {
if (!events.isEmpty()) {
eventsCopy = events;
events = new ArrayList();
currentTry = 1;
}
if (eventsCopy == null) {
if (!outputAggregator.hasPendingData()) {
// all done
working = false;
return;
} else {
// Nothing new, must be a retry
if (currentTry > maxRetries) {
// Oh well, better luck next time
working = false;
return;
}
}
}
}
if (eventsCopy != null) {
serializeEvents(jsonGenerator, eventsCopy, propertyList);
}
if (!outputAggregator.sendData()) {
currentTry++;
}
} catch (Exception e) {
errorReporter.logError("Internal error handling log data: " + e.getMessage(), e);
currentTry++;
}
}
}
private void serializeEvents(JsonGenerator gen, List eventsCopy, List> propertyList) throws IOException {
for (T event : eventsCopy) {
serializeIndexString(gen, event);
gen.writeRaw('\n');
serializeEvent(gen, event, propertyList);
gen.writeRaw('\n');
}
gen.flush();
}
private void serializeIndexString(JsonGenerator gen, T event) throws IOException {
gen.writeStartObject();
gen.writeObjectFieldStart("index");
gen.writeObjectField("_index", indexPattern.encode(event));
String type = settings.getType();
if (type != null) {
gen.writeObjectField("_type", type);
}
gen.writeEndObject();
gen.writeEndObject();
}
private void serializeEvent(JsonGenerator gen, T event, List> propertyList) throws IOException {
gen.writeStartObject();
serializeCommonFields(gen, event);
for (AbstractPropertyAndEncoder pae : propertyList) {
String value = pae.encode(event);
if (pae.allowEmpty() || (value != null && !value.isEmpty())) {
gen.writeObjectField(pae.getName(), value);
}
}
gen.writeEndObject();
}
protected abstract void serializeCommonFields(JsonGenerator gen, T event) throws IOException;
protected static String getTimestamp(long timestamp) {
Calendar cal = Calendar.getInstance();
cal.setTimeInMillis(timestamp);
return DatatypeConverter.printDateTime(cal);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy