skywalking-agent初始化流程(二)-基于byteb

2020-08-10  本文已影响0人  李亚林1990

接上篇:
对于所有的skywalking-plugin.def配置文件中定义的agent插件,将通过如下代码生效匹配规则和拦截逻辑。

agentBuilder
    .type(pluginFinder.buildMatch())//匹配规则:待增强的类
    .transform(new Transformer(pluginFinder))//根据匹配规则和增强逻辑创建Transformer:
    .with(AgentBuilder.RedefinitionStrategy.RETRANSFORMATION)
    .with(new Listener())//回调接口,用来在debug模式下保存增强后的字节码文件
    .installOn(instrumentation);

1.pluginFinder.buildMatch():匹配规则

//直接根据类名称匹配的插件
private final Map<String, LinkedList<AbstractClassEnhancePluginDefine>> nameMatchDefine = new HashMap<String, LinkedList<AbstractClassEnhancePluginDefine>>();
//非名称匹配:方法注解匹配、类注解匹配等
private final List<AbstractClassEnhancePluginDefine> signatureMatchDefine = new ArrayList<AbstractClassEnhancePluginDefine>();
//返回bytebuddy元素匹配器,用户判断是否匹配
public ElementMatcher<? super TypeDescription> buildMatch() {
        ElementMatcher.Junction judge = new AbstractJunction<NamedElement>() {
            @Override
            public boolean matches(NamedElement target) {
                //直接根据类名称匹配
                return nameMatchDefine.containsKey(target.getActualName());
            }
        };
        judge = judge.and(not(isInterface()));
        for (AbstractClassEnhancePluginDefine define : signatureMatchDefine) {
            ClassMatch match = define.enhanceClass();
            if (match instanceof IndirectMatch) {
                //其他规则匹配
                judge = judge.or(((IndirectMatch)match).buildJunction());
            }
        }
        return new ProtectiveShieldMatcher(judge);
    }

增强类匹配规则需要实现ClassMatch接口,skywalking中预定义了一些常用匹配规则
ER图如下:


image.png

NameMatch:直接根据类名称匹配
ClassAnnotationMatch:类注解匹配
MethodAnnotationMatch:方法注解匹配
HierarchyMatch:父类或父接口匹配
MultiClassNameMatch:直接根据类名称匹配(类名称数组、逻辑关系--或)

2.new Transformer(pluginFinder):根据匹配规则和增强逻辑创建AgentBuilder.Transformer

private static class Transformer implements AgentBuilder.Transformer {
        private PluginFinder pluginFinder;

        Transformer(PluginFinder pluginFinder) {
            this.pluginFinder = pluginFinder;
        }

        @Override
        public DynamicType.Builder<?> transform(DynamicType.Builder<?> builder, TypeDescription typeDescription,
            ClassLoader classLoader, JavaModule module) {
            //当前类typeDescription对应的插件集合
            List<AbstractClassEnhancePluginDefine> pluginDefines = pluginFinder.find(typeDescription);
            if (pluginDefines.size() > 0) {
                DynamicType.Builder<?> newBuilder = builder;
                EnhanceContext context = new EnhanceContext();
                for (AbstractClassEnhancePluginDefine define : pluginDefines) {
                    //插件对应的增强逻辑
                    DynamicType.Builder<?> possibleNewBuilder = define.define(typeDescription, newBuilder, classLoader, context);
                    if (possibleNewBuilder != null) {
                        newBuilder = possibleNewBuilder;
                    }
                }
                if (context.isEnhanced()) {
                    logger.debug("Finish the prepare stage for {}.", typeDescription.getName());
                }

                return newBuilder;
            }
            return builder;
        }
    }

此处需要重点解释的是如何基于skywaling插件定义的增强逻辑构建bytebuddy的DynamicType.Builder(动态类型构造器)
回顾上篇的插件定义ER图:


image.png

依次次通过如下函数链构建DynamicType.Builder
AbstractClassEnhancePluginDefine.define => ClassEnhancePluginDefine.enhance
=>ClassEnhancePluginDefine.enhanceClass=>ClassEnhancePluginDefine.enhanceInstance

@Override
    protected DynamicType.Builder<?> enhance(TypeDescription typeDescription,
        DynamicType.Builder<?> newClassBuilder, ClassLoader classLoader,
        EnhanceContext context) throws PluginException {
        //根据getStaticMethodsInterceptPoints(待增强的静态方法和拦截处理器)
        //在DynamicType.Builder构建中嵌入增强逻辑
        newClassBuilder = this.enhanceClass(typeDescription, newClassBuilder, classLoader);
        
        //根据getConstructorsInterceptPoints(待增强的构造方法和拦截处理器)
        //根据getInstanceMethodsInterceptPoints(待增强的实例方法和拦截处理器)
        //在DynamicType.Builder构建中嵌入增强逻辑
        newClassBuilder = this.enhanceInstance(typeDescription, newClassBuilder, classLoader, context);

        return newClassBuilder;
    }

以静态方法增强逻辑嵌入为例子:

StaticMethodsInterceptPoint[] staticMethodsInterceptPoints = getStaticMethodsInterceptPoints();
for (StaticMethodsInterceptPoint staticMethodsInterceptPoint : staticMethodsInterceptPoints) {
            String interceptor = staticMethodsInterceptPoint.getMethodsInterceptor();
            newClassBuilder =         
//指定方法匹配规则  
newClassBuilder.method(isStatic().and(staticMethodsInterceptPoint.getMethodsMatcher()))
                        .intercept(
                            MethodDelegation.withDefaultConfiguration()
                                .withBinders(
                                    Morph.Binder.install(OverrideCallable.class)
                                )
                                //拦截处理器
                                .to(new StaticMethodsInterWithOverrideArgs(interceptor))
                        );

3.new Listener() 回调接口,用来在debug模式下保存增强后的字节码文件
实现bytebuddy的AgentBuilder.Listener接口,基于事件回调

private static class Listener implements AgentBuilder.Listener {
        @Override
        public void onDiscovery(String typeName, ClassLoader classLoader, JavaModule module, boolean loaded) {

        }

        @Override
        public void onTransformation(TypeDescription typeDescription, ClassLoader classLoader, JavaModule module,
            boolean loaded, DynamicType dynamicType) {
            if (logger.isDebugEnable()) {
                logger.debug("On Transformation class {}.", typeDescription.getName());
            }
            //在debug模式下保存增强后的字节码文件
            InstrumentDebuggingClass.INSTANCE.log(dynamicType);
        }

        @Override
        public void onIgnored(TypeDescription typeDescription, ClassLoader classLoader, JavaModule module,
            boolean loaded) {

        }

        @Override
        public void onError(String typeName, ClassLoader classLoader, JavaModule module, boolean loaded,
            Throwable throwable) {
            logger.error("Enhance class " + typeName + " error.", throwable);
        }

        @Override
        public void onComplete(String typeName, ClassLoader classLoader, JavaModule module, boolean loaded) {
        }
    }
上一篇 下一篇

猜你喜欢

热点阅读