SpringCloudGateway使用Skywalking时日志打印traceId

前置信息

Skywalking oap 与 agent部署
https://blog.csdn.net/kismet2399/article/details/131560171

环境配置

  • spring-cloud-starter-gateway:3.1.4
  • Skywalking Agent:8.14.0

背景

SpringCloudGateway集成Skywalking后无法打印traceId

解决方式

目前没有找到logback的解决方式,所以日志打印改用log4j2

  • mvn添加配置
   <dependency>
            <groupId>org.apache.skywalking</groupId>
            <artifactId>apm-toolkit-log4j-2.x</artifactId>
            <version>8.14.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.skywalking</groupId>
            <artifactId>apm-toolkit-trace</artifactId>
            <version>8.14.0</version>
        </dependency>
          <!--解决log4j2依赖问题-->
         <dependency>
            <groupId>com.lmax</groupId>
            <artifactId>disruptor</artifactId>
            <version>3.4.4</version>
        </dependency>
  • 让日志获取到traceId
import org.springframework.stereotype.Component;
import reactor.core.publisher.Hooks;
import reactor.core.publisher.Operators;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

/**
 * @author kismet
 */
@Component
public class LogHooks {

    private static final String KEY = "logMdc";

    @PostConstruct
    public void setHook() {
        Hooks.onEachOperator(KEY,
                Operators.lift((scannable, coreSubscriber) -> new MdcSubscriber(coreSubscriber)));
    }

    @PreDestroy
    public void resetHook() {
        Hooks.resetOnEachOperator(KEY);
    }

}

import com.alibaba.fastjson.JSON;
import com.alibaba.nacos.common.utils.JacksonUtils;
import org.jboss.logging.MDC;
import org.reactivestreams.Subscription;
import reactor.core.CoreSubscriber;
import reactor.util.context.Context;

import java.lang.reflect.Field;
import java.util.Optional;

/**
 * @author zuozhuan
 */
public class MdcSubscriber implements CoreSubscriber {

    private static final String TRACE_ID = "traceId";

    private static final String SKYWALKING_CTX_SNAPSHOT = "SKYWALKING_CONTEXT_SNAPSHOT";

    private final CoreSubscriber<Object> actual;

    public MdcSubscriber(CoreSubscriber<Object> actual) {
        this.actual = actual;
    }

    @Override
    public void onSubscribe(Subscription s) {
        actual.onSubscribe(s);
    }

    @Override
    public void onNext(Object o) {
        Context c = actual.currentContext();
        Optional<String> traceIdOptional = Optional.empty();
        if (!c.isEmpty() && c.hasKey(SKYWALKING_CTX_SNAPSHOT)) {
            Object object = c.get(SKYWALKING_CTX_SNAPSHOT);
            Object traceId = findField(object, "traceId");
            String ids = JacksonUtils.toJson(traceId);
            traceIdOptional = Optional.ofNullable(ids)
                    .map(JSON::parseObject)
                    .map(t -> t.get("id"))
                    .map(Object::toString);
        }

        MDC.put(TRACE_ID, traceIdOptional.orElse("N/A"));
        actual.onNext(o);
    }

    private static Object findField(Object object, String field) {
        if (object == null) {
            return null;
        }
        try {
            Class<?> aClass = object.getClass();

            Field mValuesField = null;
            mValuesField = aClass.getDeclaredField(field);
            // 3、打开访问权限
            mValuesField.setAccessible(true);
            // 4、获取 memberValuesMap
            return mValuesField.get(object);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    @Override
    public void onError(Throwable throwable) {
        actual.onError(throwable);
    }

    @Override
    public void onComplete() {
        actual.onComplete();
    }

    @Override
    public Context currentContext() {
        return actual.currentContext();
    }
  • 插件添加
    将skywalking-agent下optional-plugins中的apm-spring-cloud-gateway-3.x-plugin-8.14.0.jar和apm-spring-webflux-5.x-plugin-8.14.0.jar移动到plugins下

  • log4j2.xml配置示例

<?xml version="1.0" encoding="UTF-8"?>
<Configuration>
    <Properties>
        <Property name="PATTERN" value="%d{yyyy-MM-dd HH:mm:ss.SSS} [TID:%equals{%X{traceId}}{}{N/A}] [%thread] %-5level %logger{20}  %msg%n"/>
    </Properties>
    <Appenders>
        <Console name="Console" target="SYSTEM_OUT">
            <PatternLayout pattern="${PATTERN}"/>
        </Console>
        <Async name="Async">
            <AppenderRef ref="Console"/>
        </Async>
        <!--输出到日志文件,滚动分割日志文件,自动打包gz -->
        <RollingFile name="INFO_FILE" fileName="${env:HOME}/logs/gateway/gateway.log" filePattern="${env:HOME}/logs/gateway/gateway-%d{yyyyMMdd}.log.%i">
            <PatternLayout pattern="${PATTERN}"/>
            <Policies>
                <!--默认一天一个文件 -->
                <TimeBasedTriggeringPolicy />
                <!--一天内大于size就单独分隔-->
                <SizeBasedTriggeringPolicy size="500MB"/>
            </Policies>
        </RollingFile>
        <!--skywalking 日志收集 -->
        <GRPCLogClientAppender name="APM_LOG">
            <PatternLayout pattern="${PATTERN}"/>
        </GRPCLogClientAppender>
    </Appenders>

    <Loggers>
        <Root level="INFO">
            <AppenderRef ref="Console">
                <Filters>
                    <PropertyFilter key="spring.profiles.active" value="dev" onMatch="ACCEPT" onMismatch="DENY" />
                </Filters>
            </AppenderRef>
            <AppenderRef ref="INFO_FILE"/>
            <AppenderRef ref="APM_LOG"/>
        </Root>
    </Loggers>
</Configuration>

  • 如果使用skywalking 日志收集启动参数需添加启动参数
-Dlog4j2.contextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector

参考链接:
https://www.jianshu.com/p/40727a0b9604
https://skywalking.apache.org/docs/skywalking-java/v8.14.0/en/setup/service-agent/java-agent/application-toolkit-log4j-2.x/

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
SpringCloud是一个开源的分布式系统开发框架,可以用于快速构建基于Java的微服务架构。而SkyWalking是一个开源的分布式APM(Application Performance Management)系统,用于监控、诊断和管理分布式系统的性能。 要实现SpringCloud服务整合SkyWalking,首先我们需要在SpringCloud微服务架构集成SkyWalking的Agent组件。通过在每个微服务应用引入SkyWalking的Agent,就可以实现对每个微服务节点的性能监控。其次,我们还需要在SkyWalking的Server端配置和管理各个微服务应用的监控数据,以便进行统一的性能分析和故障定位。 在服务整合的过程,还需要考虑一些细节问题,比如Agent的版本兼容性、监控数据的采集频率和存储策略、以及监控数据的可视化展示等。同,还需要注意Agent的性能开销和系统资源消耗,避免给微服务应用带来过大的性能损耗。 通过整合SpringCloudSkyWalking,我们可以实现对整个微服务架构的性能监控和分析,及发现系统的性能瓶颈和故障节点,为系统的稳定性和可靠性提供保障。同,还可以利用SkyWalking的可视化监控界面,直观地展示微服务架构的性能指标和运行情况,为开发人员和运维人员提供更加直观和有效的管理工具。 总之,SpringCloud服务整合SkyWalking是一个非常有益的举措,可以提升微服务架构的可观测性和可管理性,为系统的稳定性和高效性提供保障。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值