All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.irain.reactor.logging.appender.LoggingLogStashAppender Maven / Gradle / Ivy

The newest version!
package io.irain.reactor.logging.appender;

import ch.qos.logback.classic.Logger;
import ch.qos.logback.classic.LoggerContext;
import cn.hutool.json.JSONUtil;
import io.irain.reactor.logging.properties.LoggingProperties;
import io.irain.reactor.logging.utils.LogStashUtil;
import lombok.extern.slf4j.Slf4j;
import net.logstash.logback.appender.LogstashTcpSocketAppender;
import net.logstash.logback.encoder.LogstashEncoder;
import org.slf4j.LoggerFactory;
import org.springframework.core.env.Environment;

import java.util.HashMap;
import java.util.Map;

/**
 * LogStash 输出 json
 *
 * @author youta
 */
@Slf4j
public class LoggingLogStashAppender implements ILoggingAppender {
	private static final String ASYNC_LOG_STASH_APPENDER_NAME = "ASYNC_LOG_STASH";
	private final LoggingProperties properties;
	private final String customFieldsJson;

	/**
	 * 构造器
	 * @param environment environment
	 * @param properties properties
	 */
	public LoggingLogStashAppender(Environment environment,
								   LoggingProperties properties) {
		this.properties = properties;
		// 1. 服务名和环境
		String appName = environment.getRequiredProperty("spring.application.name");
		String profile = environment.getRequiredProperty("spring.profiles.active");
		// 2. json 自定义字段
		Map customFields = new HashMap<>(4);
		customFields.put("appName", appName);
		customFields.put("profile", profile);
		// 3. 自定义配置的字段
		customFields.putAll(properties.getLogstash().getCustomFieldMap());
		this.customFieldsJson = JSONUtil.toJsonStr(customFields);
		LoggerContext context = (LoggerContext) LoggerFactory.getILoggerFactory();
		this.start(context);
	}

	@Override
	public void start(LoggerContext context) {
		log.info("LogStash logging start.");
		reload(context);
	}

	@Override
	public void reset(LoggerContext context) {
		log.info("LogStash logging reset.");
		reload(context);
	}

	private void reload(LoggerContext context) {
		LoggingProperties.Logstash logStash = properties.getLogstash();
		if (logStash.isEnabled()) {
			addLogStashTcpSocketAppender(context, customFieldsJson, logStash);
		}
	}

	/**
	 * addLogstashTcpSocketAppender.
	 *
	 * @param context            a {@link LoggerContext} object.
	 * @param customFields       a {@link String} object.
	 * @param logStashProperties a {@link LoggingProperties.Logstash} object.
	 */
	private static void addLogStashTcpSocketAppender(LoggerContext context,
													String customFields,
													LoggingProperties.Logstash logStashProperties) {
		// More documentation is available at: https://github.com/logstash/logstash-logback-encoder
		final LogstashTcpSocketAppender logStashAppender = new LogstashTcpSocketAppender();
		logStashAppender.addDestination(logStashProperties.getDestinations());
		logStashAppender.setContext(context);
		logStashAppender.setEncoder(logstashEncoder(customFields));
		logStashAppender.setName(ASYNC_LOG_STASH_APPENDER_NAME);
		logStashAppender.setRingBufferSize(logStashProperties.getRingBufferSize());
		logStashAppender.start();
		// 先删除,再添加
		context.getLogger(Logger.ROOT_LOGGER_NAME).detachAppender(ASYNC_LOG_STASH_APPENDER_NAME);
		context.getLogger(Logger.ROOT_LOGGER_NAME).addAppender(logStashAppender);
	}

	private static LogstashEncoder logstashEncoder(String customFields) {
		final LogstashEncoder logstashEncoder = new LogstashEncoder();
		logstashEncoder.setThrowableConverter(LogStashUtil.throwableConverter());
		logstashEncoder.setCustomFields(customFields);
		return logstashEncoder;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy