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

fun.fengwk.convention4j.tracer.scope.TtlScopeManager Maven / Gradle / Ivy

package fun.fengwk.convention4j.tracer.scope;

import com.alibaba.ttl.TransmittableThreadLocal;
import com.google.auto.service.AutoService;
import fun.fengwk.convention4j.tracer.util.TracerUtils;
import io.opentracing.Scope;
import io.opentracing.Span;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.ToString;
import lombok.extern.slf4j.Slf4j;

import java.util.LinkedList;
import java.util.Map;

/**
 * @author fengwk
 */
@AutoService(ConventionScopeManager.class)
@Slf4j
public class TtlScopeManager implements ConventionScopeManager {

    private static final int MAX_STACK_SIZE = 1000;
    private static final TransmittableThreadLocal> TTL_SCOPE_STACK = new TransmittableThreadLocal<>() {
        @Override
        protected LinkedList initialValue() {
            return new LinkedList<>();
        }

        @Override
        protected void beforeExecute() {
            // 使用自定义agent替代
//            super.beforeExecute();
//            LinkedList scopeStack = get();
//            synchronized (scopeStack) {
//                if (!scopeStack.isEmpty()) {
//                    ScopeImpl scope = scopeStack.getFirst();
//                    TracerUtils.setMDC(scope.getSpan().context());
//                }
//            }
        }

        @Override
        protected void afterExecute() {
            // 使用自定义agent替代
//            super.afterExecute();
//            LinkedList scopeStack = get();
//            synchronized (scopeStack) {
//                if (!scopeStack.isEmpty()) {
//                    ScopeImpl scope = scopeStack.getFirst();
//                    TracerUtils.clearMDC(scope.getSpan().context());
//                }
//            }
        }
    };

    private static Scope doActive(Span span) {
        LinkedList scopeStack = TTL_SCOPE_STACK.get();
        Map storeMdc = TracerUtils.setMDC(span.context());
        ScopeImpl scope = new ScopeImpl(scopeStack, span, storeMdc);
        synchronized (scopeStack) {
            // addFirst可以使最近添加的节点以最少的遍历数被移出LinkedList
            scopeStack.addFirst(scope);
            // 防止编程错误导致的内存泄露
            if (scopeStack.size() > MAX_STACK_SIZE) {
                log.error("Scope stack size exceeds max stack size: {}", MAX_STACK_SIZE);
                ScopeImpl removedScope = scopeStack.removeLast();
                removedScope.close();
            }
        }
        return scope;
    }

    @Override
    public Scope activate(Span span) {
        return doActive(span);
    }

    @Override
    public Span activeSpan() {
        LinkedList scopeStack = TTL_SCOPE_STACK.get();
        synchronized (scopeStack) {
            return scopeStack.isEmpty() ? null : scopeStack.getFirst().getSpan();
        }
    }

    @Override
    public void close() {
        TTL_SCOPE_STACK.remove();
    }

    @EqualsAndHashCode
    @ToString
    public static class ScopeImpl implements Scope {

        private final LinkedList scopeStack;
        @Getter
        private final Span span;
        private final Map storeMdc;

        public ScopeImpl(LinkedList scopeStack, Span span, Map storeMdc) {
            this.scopeStack = scopeStack;
            this.span = span;
            this.storeMdc = storeMdc;
        }

        @Override
        public void close() {
            synchronized (scopeStack) {
                if (scopeStack.remove(this)) {
                    TracerUtils.clearMDC(storeMdc);
                }
            }
        }

    }

    @Data
    static class TraceInfo {
        private final String traceId;
        private final String spanId;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy