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

ai.databand.azkaban.agent.AzkabanTransformer Maven / Gradle / Ivy

/*
 * © Copyright Databand.ai, an IBM Company 2022
 */

package ai.databand.azkaban.agent;

import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.LoaderClassPath;
import javassist.bytecode.MethodInfo;

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.instrument.ClassFileTransformer;
import java.security.ProtectionDomain;
import java.util.Optional;

public class AzkabanTransformer implements ClassFileTransformer {

    private static final String TASK_ANNOTATION = "ai.databand.annotations.Task";

    private final boolean isVerbose;

    public AzkabanTransformer(boolean isVerbose) {
        this.isVerbose = isVerbose;
    }

    public byte[] transform(ClassLoader loader,
                            String className,
                            Class classBeingRedefined,
                            ProtectionDomain protectionDomain,
                            byte[] classfileBuffer) {
        ClassPool cp = ClassPool.getDefault();
        cp.appendClassPath(new LoaderClassPath(loader));
        Optional ctOpt = classInScope(cp, className, classfileBuffer);

        if (!ctOpt.isPresent()) {
            return null;
        }

        try {
            CtClass ct = ctOpt.get();
            System.out.printf("Instrumenting class %s%n", className);
            CtMethod[] declaredMethods = ct.getDeclaredMethods();

            for (CtMethod method : declaredMethods) {
                // wrap methods annotated by @task
                MethodInfo methodInfo = method.getMethodInfo();

                if (method.getName().contains("handleEvent")) {
                    if (isVerbose) {
                        System.out.printf("Instrumenting method %s%n", methodInfo.getName());
                    }

                    method.insertBefore("{ new ai.databand.azkaban.DbndEventReporter().report($1); }");
                    break;
                }
            }

            return ct.toBytecode();
        } catch (RuntimeException e) {
            if (e.getMessage() != null && e.getMessage().contains("frozen")) {
                return null;
            }
        } catch (Throwable e) {
            e.printStackTrace();
        }

        return classfileBuffer;
    }

    protected Optional classInScope(ClassPool cp, String className, byte[] classfileBuffer) {
        if (!className.equalsIgnoreCase("azkaban/execapp/FlowRunnerManager")
            && !className.contains("azkaban/execapp/FlowRunner$JobRunnerEventListener")) {
            return Optional.empty();
        }
        try (InputStream is = new ByteArrayInputStream(classfileBuffer)) {
            CtClass ct = cp.makeClass(is);
            return Optional.of(ct);
        } catch (IOException e) {
            return Optional.empty();
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy