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

io.automatiko.engine.quarkus.deployment.AutomatikoQuarkusProcessor Maven / Gradle / Ivy

The newest version!
package io.automatiko.engine.quarkus.deployment;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.net.URI;
import java.net.URL;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Scanner;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.Collectors;

import javax.tools.Diagnostic;
import javax.tools.DiagnosticCollector;
import javax.tools.JavaCompiler;
import javax.tools.JavaCompiler.CompilationTask;
import javax.tools.JavaFileObject;
import javax.tools.SimpleJavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.StandardLocation;
import javax.tools.ToolProvider;

import org.eclipse.microprofile.config.ConfigProvider;
import org.jboss.jandex.AnnotationInstance;
import org.jboss.jandex.ClassInfo;
import org.jboss.jandex.CompositeIndex;
import org.jboss.jandex.DotName;
import org.jboss.jandex.Index;
import org.jboss.jandex.IndexView;
import org.jboss.jandex.Indexer;
import org.jboss.jandex.MethodInfo;
import org.jboss.jandex.Type;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import io.automatiko.engine.api.Model;
import io.automatiko.engine.api.codegen.AutomatikoConfigProperties;
import io.automatiko.engine.api.definition.process.Process;
import io.automatiko.engine.codegen.ApplicationGenerator;
import io.automatiko.engine.codegen.CodeGenConstants;
import io.automatiko.engine.codegen.GeneratedFile;
import io.automatiko.engine.codegen.GeneratorContext;
import io.automatiko.engine.codegen.LambdaParser;
import io.automatiko.engine.codegen.context.QuarkusApplicationBuildContext;
import io.automatiko.engine.codegen.decision.DecisionCodegen;
import io.automatiko.engine.codegen.di.CDIDependencyInjectionAnnotator;
import io.automatiko.engine.codegen.process.ProcessCodegen;
import io.automatiko.engine.codegen.process.persistence.PersistenceGenerator;
import io.automatiko.engine.quarkus.AutomatikoBuildTimeConfig;
import io.automatiko.engine.services.utils.IoUtils;
import io.automatiko.engine.workflow.BaseWorkItem;
import io.automatiko.engine.workflow.builder.BuilderContext;
import io.automatiko.engine.workflow.builder.WorkflowBuilder;
import io.automatiko.engine.workflow.marshalling.impl.AutomatikoMessages;
import io.automatiko.engine.workflow.process.executable.core.ExecutableProcess;
import io.quarkus.arc.deployment.GeneratedBeanBuildItem;
import io.quarkus.bootstrap.classloading.ClassPathElement;
import io.quarkus.bootstrap.classloading.MemoryClassPathElement;
import io.quarkus.bootstrap.classloading.QuarkusClassLoader;
import io.quarkus.bootstrap.model.ApplicationModel;
import io.quarkus.deployment.ApplicationArchive;
import io.quarkus.deployment.Capabilities;
import io.quarkus.deployment.annotations.BuildProducer;
import io.quarkus.deployment.annotations.BuildStep;
import io.quarkus.deployment.builditem.AdditionalIndexedClassesBuildItem;
import io.quarkus.deployment.builditem.ApplicationArchivesBuildItem;
import io.quarkus.deployment.builditem.ArchiveRootBuildItem;
import io.quarkus.deployment.builditem.FeatureBuildItem;
import io.quarkus.deployment.builditem.GeneratedResourceBuildItem;
import io.quarkus.deployment.builditem.LaunchModeBuildItem;
import io.quarkus.deployment.builditem.LiveReloadBuildItem;
import io.quarkus.deployment.builditem.nativeimage.NativeImageResourceBuildItem;
import io.quarkus.deployment.builditem.nativeimage.ReflectiveClassBuildItem;
import io.quarkus.deployment.builditem.nativeimage.ReflectiveHierarchyBuildItem;
import io.quarkus.deployment.builditem.nativeimage.ReflectiveHierarchyIgnoreWarningBuildItem;
import io.quarkus.deployment.builditem.nativeimage.RuntimeInitializedClassBuildItem;
import io.quarkus.deployment.builditem.nativeimage.ServiceProviderBuildItem;
import io.quarkus.deployment.dev.testing.TestListener;
import io.quarkus.deployment.dev.testing.TestListenerBuildItem;
import io.quarkus.deployment.dev.testing.TestRunListener;
import io.quarkus.deployment.index.IndexingUtil;
import io.quarkus.deployment.pkg.PackageConfig;
import io.quarkus.deployment.pkg.builditem.CurateOutcomeBuildItem;
import io.quarkus.maven.dependency.ResolvedDependency;
import io.quarkus.paths.PathCollection;
import io.quarkus.runtime.LaunchMode;

public class AutomatikoQuarkusProcessor {

    public static final String DEFAULT_PACKAGE_NAME = "io.automatiko.app";

    private static final String FEATURE = "automatiko";

    private static final String generatedResourcesDir = System.getProperty("io.automatiko.codegen.resources.directory",
            "target/generated-resources/automatiko");

    private static final String generatedSourcesDir = "target/generated-sources/automatiko/";
    private static final String generatedTestSourcesDir = "target/generated-test-sources/automatiko/";

    private static final Logger logger = LoggerFactory.getLogger(AutomatikoQuarkusProcessor.class);
    private final transient String generatedClassesDir = System.getProperty("quarkus.debug.generated-classes-dir");
    private final transient String persistenceFactoryClass = "io.automatiko.engine.addons.persistence.AbstractProcessInstancesFactory";

    private final transient String workflowsAnnotationClass = "io.automatiko.engine.api.Workflows";
    private final transient String workflowsBuilderClass = "io.automatiko.engine.workflow.builder.WorkflowBuilder";

    @BuildStep
    public void generateClassesFromResourcesStep(AutomatikoBuildTimeConfig config, PackageConfig pconfig,
            ArchiveRootBuildItem root,
            ApplicationArchivesBuildItem archives,
            LaunchModeBuildItem launchMode,
            LiveReloadBuildItem liveReload,
            CurateOutcomeBuildItem curateOutcomeBuildItem,
            Capabilities capabilities,
            BuildProducer generatedBeans,
            BuildProducer additionalIndexClass,
            BuildProducer resource,
            BuildProducer reflectiveClass,
            BuildProducer resources,
            BuildProducer providerProducer,
            BuildProducer testListenerProducer)
            throws Exception {

        testListenerProducer.produce(new TestListenerBuildItem(new TestListener() {

            @Override
            public void testsEnabled() {

                System.setProperty("test.runs.enabled", "true");
            }

            @Override
            public void testRunStarted(Consumer listenerConsumer) {
                System.setProperty("test.runs.enabled", "true");
            }

        }));
        // prepare index
        List archiveIndexes = new ArrayList<>();

        for (ApplicationArchive i : archives.getAllApplicationArchives()) {
            archiveIndexes.add(i.getIndex());
        }

        CompositeIndex archivesIndex = CompositeIndex.create(archiveIndexes);
        boolean onlyCodeAssets = false;
        if (liveReload.isLiveReload() || ConfigProvider.getConfig()
                .getOptionalValue("quarkus.live-reload.url", String.class).isPresent()
                || "true".equals(System.getProperty("test.runs.enabled"))) {

            List workflowBuilderClasses = archivesIndex
                    .getAnnotations(createDotName(workflowsAnnotationClass)).stream()
                    .map(a -> a.target().asClass().name().toString()).collect(Collectors.toList());

            if (liveReload.getChangeInformation() != null && (liveReload.getChangeInformation().getAddedClasses().stream()
                    .anyMatch(clazz -> workflowBuilderClasses.contains(clazz))
                    || liveReload.getChangeInformation().getChangedClasses().stream()
                            .anyMatch(clazz -> workflowBuilderClasses.contains(clazz)))) {
                onlyCodeAssets = true;
            } else {
                return;
            }
        }

        AppPaths appPaths = new AppPaths(root.getResolvedPaths());

        ApplicationGenerator appGen = createApplicationGenerator(capabilities, config, appPaths, archivesIndex,
                curateOutcomeBuildItem.getApplicationModel(), onlyCodeAssets);

        Collection generatedFiles = appGen.generate();

        Collection javaFiles = generatedFiles.stream().filter(f -> f.relativePath().endsWith(".java"))
                .collect(Collectors.toCollection(ArrayList::new));
        String sourceFolder = sourceFolder(appGen);
        writeGeneratedFiles(appGen, sourceFolder, appPaths, generatedFiles);

        if (!javaFiles.isEmpty()) {

            Indexer automatikoIndexer = new Indexer();
            Set automatikoIndex = new HashSet<>();

            compile(appGen, appPaths, curateOutcomeBuildItem.getApplicationModel(), javaFiles, launchMode.getLaunchMode(),
                    generatedBeans, additionalIndexClass, (className, data) -> {
                        return generateBeanBuildItem(archivesIndex, automatikoIndexer, automatikoIndex, className, data);
                    }, pconfig);

            Index index = automatikoIndexer.complete();

            Collection dataEvents = index
                    .getAllKnownSubclasses(createDotName("io.automatiko.engine.api.event.AbstractDataEvent"));

            dataEvents.forEach(
                    c -> {
                        reflectiveClass
                                .produce(ReflectiveClassBuildItem.builder(c.name().toString()).fields().methods().build());
                    });

            generatePersistenceInfo(appGen, config, pconfig, appPaths, generatedBeans, additionalIndexClass,
                    CompositeIndex.create(archivesIndex, index), launchMode, resource,
                    curateOutcomeBuildItem);

        }
    }

    @BuildStep
    FeatureBuildItem feature() {
        return new FeatureBuildItem(FEATURE);
    }

    @BuildStep
    public void reflectiveClassesRegistrationStep(ApplicationArchivesBuildItem archives,
            BuildProducer reflectiveHierarchy,
            BuildProducer reflectiveClass,
            BuildProducer reflectiveClassHierarchy) {

        reflectiveClass.produce(ReflectiveClassBuildItem.builder("io.automatiko.engine.api.event.AbstractDataEvent").fields()
                .methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem.builder("io.automatiko.engine.services.event.AbstractProcessDataEvent")
                .fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem.builder("io.automatiko.engine.services.event.ProcessInstanceDataEvent")
                .fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.UserTaskInstanceDataEvent").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.VariableInstanceDataEvent").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.impl.ProcessInstanceEventBody").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.impl.NodeInstanceEventBody").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.impl.ProcessErrorEventBody").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.impl.VariableInstanceEventBody").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem
                .builder("io.automatiko.engine.services.event.impl.UserTaskInstanceEventBody").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("org.mvel2.optimizers.dynamic.DynamicOptimizer").fields().methods().build());
        reflectiveClass.produce(ReflectiveClassBuildItem.builder("org.mvel2.optimizers.impl.refl.ReflectiveAccessorOptimizer")
                .fields().methods().build());
        reflectiveClass
                .produce(ReflectiveClassBuildItem.builder(ArrayList.class.getCanonicalName()).fields().methods().build());

        reflectiveClass.produce(ReflectiveClassBuildItem.builder(byte[].class.getName()).fields().methods().build());

        reflectiveClass.produce(ReflectiveClassBuildItem.builder(BaseWorkItem.class.toString()).fields().methods().build());

        List archiveIndexes = new ArrayList<>();

        for (ApplicationArchive i : archives.getAllApplicationArchives()) {
            archiveIndexes.add(i.getIndex());
        }

        CompositeIndex archivesIndex = CompositeIndex.create(archiveIndexes);

        // add functions classes found
        reflectiveClass.produce(ReflectiveClassBuildItem.builder("io.automatiko.engine.services.execution.BaseFunctions")
                .fields().methods().build());
        archivesIndex
                .getAllKnownImplementors(createDotName("io.automatiko.engine.api.Functions")).stream()
                .map(c -> c.name().toString())
                .forEach(
                        c -> reflectiveClass.produce(ReflectiveClassBuildItem.builder(c).methods().build()));

        // DMN related
        DotName classDotName = createDotName(CodeGenConstants.DMN_CLASS);
        if (!archivesIndex.getAnnotations(classDotName).isEmpty() || archivesIndex.getClassByName(classDotName) != null) {
            reflectiveHierarchy
                    .produce(new ReflectiveHierarchyIgnoreWarningBuildItem(createDotName("org.kie.api.builder.Message$Level")));
            reflectiveHierarchy
                    .produce(new ReflectiveHierarchyIgnoreWarningBuildItem(createDotName("org.kie.dmn.api.core.DMNContext")));
            reflectiveHierarchy.produce(
                    new ReflectiveHierarchyIgnoreWarningBuildItem(createDotName("org.kie.dmn.api.core.DMNDecisionResult")));
            reflectiveHierarchy.produce(new ReflectiveHierarchyIgnoreWarningBuildItem(
                    createDotName("org.kie.dmn.api.core.DMNDecisionResult$DecisionEvaluationStatus")));
            reflectiveHierarchy
                    .produce(new ReflectiveHierarchyIgnoreWarningBuildItem(createDotName("org.kie.dmn.api.core.DMNMessage")));
            reflectiveHierarchy.produce(new ReflectiveHierarchyIgnoreWarningBuildItem(
                    createDotName("org.kie.dmn.api.core.DMNMessage$Severity")));
            reflectiveHierarchy
                    .produce(new ReflectiveHierarchyIgnoreWarningBuildItem(
                            createDotName("org.kie.dmn.api.core.DMNMessageType")));
            reflectiveHierarchy.produce(new ReflectiveHierarchyIgnoreWarningBuildItem(
                    createDotName("org.kie.dmn.api.feel.runtime.events.FEELEvent")));
        }

        // jackson node classes for serializing JSON tree instances
        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.TextNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.BinaryNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.BooleanNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.NullNode").fields().methods().build());

        reflectiveClass.produce(ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.BigIntegerNode").fields()
                .methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.DecimalNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.DoubleNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.FloatNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.IntNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.LongNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.ShortNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.POJONode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.ObjectNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder("com.fasterxml.jackson.databind.node.ArrayNode").fields().methods().build());

        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder(Long.class.getCanonicalName()).fields().methods().build());
        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder(Integer.class.getCanonicalName()).fields().methods().build());
        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder(Boolean.class.getCanonicalName()).fields().methods().build());
        reflectiveClass.produce(
                ReflectiveClassBuildItem.builder(String.class.getCanonicalName()).fields().methods().build());

        try {
            Enumeration reflectionConfigs = Thread.currentThread().getContextClassLoader()
                    .getResources("/META-INF/automatiko.reflection.classes");

            while (reflectionConfigs.hasMoreElements()) {
                URL url = (URL) reflectionConfigs.nextElement();

                try (InputStream in = url.openStream(); Scanner scanner = new Scanner(in);) {

                    while (scanner.hasNextLine()) {
                        String className = scanner.nextLine();
                        reflectiveClass.produce(
                                ReflectiveClassBuildItem.builder(className).constructors().methods().fields().build());
                    }
                }
            }
        } catch (Exception e) {
            logger.warn("Unexpected error while reading reflection config files", e);
        }

        try {
            Enumeration reflectionConfigs = Thread.currentThread().getContextClassLoader()
                    .getResources("/META-INF/automatiko.reflection.interfaces");

            while (reflectionConfigs.hasMoreElements()) {
                URL url = (URL) reflectionConfigs.nextElement();

                try (InputStream in = url.openStream(); Scanner scanner = new Scanner(in);) {

                    while (scanner.hasNextLine()) {
                        String className = scanner.nextLine();

                        reflectiveClassHierarchy.produce(new ReflectiveHierarchyBuildItem.Builder()
                                .type(Type.create(DotName.createSimple(className), Type.Kind.CLASS))
                                .source(getClass().getSimpleName() + ">" + className)
                                .build());
                    }
                }
            }
        } catch (Exception e) {
            logger.warn("Unexpected error while reading reflection config files", e);
        }

    }

    @BuildStep
    public void runtimeInitializationRegistrationStep(BuildProducer producer) {

        producer.produce(new RuntimeInitializedClassBuildItem(AutomatikoMessages.class.getCanonicalName()));
    }

    @BuildStep
    public void serviceProviderRegistrationStep(AutomatikoBuildTimeConfig config,
            BuildProducer providerProducer) {

        providerProducer.produce(new ServiceProviderBuildItem(AutomatikoConfigProperties.class.getCanonicalName(),
                "io.automatiko.application.app.GeneratedAutomatikoConfigProperties"));

        providerProducer.produce(ServiceProviderBuildItem.allProvidersFromClassPath("net.thisptr.jackson.jq.Function"));
    }

    @BuildStep
    public void resourceFileRegistrationStep(BuildProducer resource) {

        resource.produce(new NativeImageResourceBuildItem("templates/default-usertask-email.html"));
        resource.produce(new NativeImageResourceBuildItem("templates/default-task-template.html"));
        resource.produce(new NativeImageResourceBuildItem("templates/not-found-template.html"));
        resource.produce(new NativeImageResourceBuildItem("automatiko-index.html"));
        resource.produce(new NativeImageResourceBuildItem("/META-INF/resources/js/automatiko-authorization.js"));
    }

    private void generatePersistenceInfo(ApplicationGenerator appGen, AutomatikoBuildTimeConfig config, PackageConfig pconfig,
            AppPaths appPaths,
            BuildProducer generatedBeans,
            BuildProducer additionalIndexClass,
            IndexView index, LaunchModeBuildItem launchMode,
            BuildProducer resource, CurateOutcomeBuildItem curateOutcomeBuildItem)
            throws Exception {

        ClassInfo persistenceClass = index.getClassByName(createDotName(persistenceFactoryClass));
        boolean usePersistence = persistenceClass != null;
        List parameters = new ArrayList<>();
        if (usePersistence) {
            for (MethodInfo mi : persistenceClass.methods()) {
                if (mi.name().equals("") && !mi.parameters().isEmpty()) {
                    parameters = mi.parameters().stream().map(p -> p.name().toString()).collect(Collectors.toList());
                    break;
                }
            }
        }

        Collection generatedFiles = getGeneratedPersistenceFiles(config, appPaths, index, usePersistence,
                parameters);

        if (!generatedFiles.isEmpty()) {
            String sourceFolder = sourceFolder(appGen);
            writeGeneratedFiles(appGen, sourceFolder, appPaths, generatedFiles);

            compile(appGen, appPaths, curateOutcomeBuildItem.getApplicationModel(), generatedFiles, launchMode.getLaunchMode(),
                    generatedBeans, additionalIndexClass, GeneratedBeanBuildItem::new, pconfig);
        }

    }

    private Collection getGeneratedPersistenceFiles(AutomatikoBuildTimeConfig config, AppPaths appPaths,
            IndexView index, boolean usePersistence, List parameters) {

        Collection modelClasses = index
                .getAllKnownImplementors(createDotName(Model.class.getCanonicalName()));

        Collection generatedFiles = new ArrayList<>();

        for (Path projectPath : appPaths.projectPaths) {
            PersistenceGenerator persistenceGenerator = new PersistenceGenerator(
                    new File(projectPath.toFile(), "target"), modelClasses, usePersistence,
                    Thread.currentThread().getContextClassLoader());
            persistenceGenerator.setDependencyInjection(new CDIDependencyInjectionAnnotator());
            persistenceGenerator.setPackageName(config.packageName.orElse(DEFAULT_PACKAGE_NAME));
            persistenceGenerator.setContext(AutomatikoBuildData.get().getGenerationContext());

            generatedFiles.addAll(persistenceGenerator.generate());
        }
        return generatedFiles;
    }

    private void writeGeneratedFiles(ApplicationGenerator appGen, String sourceFolder, AppPaths appPaths,
            Collection resourceFiles) {
        String generateToFolder = System.getProperty("automatiko.testOnly") != null ? generatedTestSourcesDir
                : generatedSourcesDir;
        for (Path projectPath : appPaths.projectPaths) {

            String resourcePath = projectPath.resolve(generatedResourcesDir).toString();
            String jsonSchemaPath = projectPath.resolve(generatedResourcesDir).resolve("jsonSchema").toString();
            String sourcePath = projectPath.resolve(generateToFolder).toString();

            for (GeneratedFile f : resourceFiles) {
                try {
                    if (f.getType() == GeneratedFile.Type.RESOURCE) {
                        writeGeneratedFile(f, sourceFolder, resourcePath);
                    } else if (f.getType() == GeneratedFile.Type.JSON_SCHEMA) {
                        writeGeneratedFile(f, sourceFolder, jsonSchemaPath);
                    } else {
                        writeGeneratedFile(f, sourceFolder, sourcePath);
                    }
                } catch (IOException e) {
                    logger.warn(String.format("Could not write file '%s'", f.toString()), e);
                }
            }
        }
    }

    private GeneratedBeanBuildItem generateBeanBuildItem(CompositeIndex archivesIndex,
            Indexer automatikIndexer, Set automatikIndex, String className, byte[] data) {
        IndexingUtil.indexClass(className, automatikIndexer, archivesIndex, automatikIndex,
                Thread.currentThread().getContextClassLoader(), data);
        return new GeneratedBeanBuildItem(className, data);
    }

    @SuppressWarnings("unchecked")
    private void compile(ApplicationGenerator appGen, AppPaths appPaths, ApplicationModel appModel,
            Collection generatedFiles,
            LaunchMode launchMode, BuildProducer generatedBeans,
            BuildProducer additionalIndexClassProducer,
            BiFunction bif, PackageConfig config) throws Exception {
        List sources = new ArrayList();
        List classpaths = new ArrayList();

        List options = new ArrayList();
        options.add("-proc:none"); // force disable annotation processing
        for (Path classPath : appPaths.classesPaths) {
            classpaths.add(classPath.toString());
        }
        if (appModel != null) {
            for (ResolvedDependency i : appModel.getRuntimeDependencies()) {
                classpaths.add(i.getResolvedPaths().getSinglePath().toAbsolutePath().toString());
            }
        }
        if (!classpaths.isEmpty()) {
            options.add("-classpath");
            options.add(classpaths.stream().collect(Collectors.joining(File.pathSeparator)));
        }

        JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
        DiagnosticCollector diagnosticsCollector = new DiagnosticCollector<>();
        StandardJavaFileManager fileManager = compiler.getStandardFileManager(diagnosticsCollector, null, null);

        try {
            File buildDir = appPaths.getFirstClassesPath().toFile();
            if (buildDir.isFile()) {
                buildDir = new File(buildDir.getParentFile(), "classes");

            }
            // if environment is configured with test only write all generated and compiled classes into the test-classes folder
            if (System.getProperty("automatiko.testOnly") != null) {
                buildDir = new File(buildDir.getParentFile(), "test-classes");
            }

            fileManager.setLocation(StandardLocation.CLASS_OUTPUT, Collections.singleton(buildDir));
            String sourceFolder = sourceFolder(appGen);
            for (GeneratedFile entry : generatedFiles) {
                String generatedClassFile = entry.relativePath().replace(sourceFolder + "/", "");
                String fileName = toRuntimeSource(sourceFolder, toClassName(generatedClassFile));

                sources.add(new SourceCode(fileName, new String(entry.contents())));

                String location = generatedClassesDir;
                if (launchMode == LaunchMode.DEVELOPMENT
                        || config.jar().type().equals(PackageConfig.JarConfig.JarType.MUTABLE_JAR)) {
                    location = Paths.get(buildDir.toString()).toString();

                }

                writeGeneratedFile(entry, sourceFolder, location);
            }

            CompilationTask task = compiler.getTask(null, fileManager, diagnosticsCollector, options, null, sources);
            boolean result = task.call();

            if (result) {
                Map classes = new LinkedHashMap();
                List classesToIndex = new ArrayList();
                Iterable compiledClasses = fileManager.list(StandardLocation.CLASS_OUTPUT, "",
                        Collections.singleton(JavaFileObject.Kind.CLASS), true);
                for (JavaFileObject jfo : compiledClasses) {

                    String clazz = jfo.getName().replace(buildDir.toString() + File.separator, "");
                    clazz = toClassName(clazz);
                    byte[] content = IoUtils.readBytesFromInputStream(jfo.openInputStream());
                    generatedBeans.produce(bif.apply(clazz, content));

                    classesToIndex.add(clazz);
                    classes.put(clazz, content);
                    classes.put(clazz.replaceAll("\\.", "/") + ".class", content);
                }

                if (Thread.currentThread().getContextClassLoader() instanceof QuarkusClassLoader) {
                    QuarkusClassLoader cl = (QuarkusClassLoader) Thread.currentThread().getContextClassLoader();

                    Field f = cl.getClass().getDeclaredField("normalPriorityElements");
                    f.setAccessible(true);
                    List element = (List) f.get(cl);

                    element.add(new MemoryClassPathElement(classes, result) {

                        @Override
                        public Path getRoot() {
                            return fileManager.getLocation(StandardLocation.CLASS_OUTPUT).iterator().next().toPath();
                        }

                    });

                    f = cl.getClass().getDeclaredField("classPathResourceIndex");
                    f.setAccessible(true);
                    f.set(cl, null);
                }

                additionalIndexClassProducer
                        .produce(new AdditionalIndexedClassesBuildItem(classesToIndex.toArray(String[]::new)));

            } else {
                List> diagnostics = diagnosticsCollector.getDiagnostics();
                String errorMessage = diagnostics.stream().map(d -> d.toString()).collect(Collectors.joining(","));

                throw new IllegalStateException(errorMessage);
            }
        } finally {
            fileManager.close();
        }
    }

    private ApplicationGenerator createApplicationGenerator(Capabilities capabilities, AutomatikoBuildTimeConfig config,
            AppPaths appPaths,
            CompositeIndex archivesIndex, ApplicationModel appModel, boolean onlyCodeAssets) throws IOException {

        boolean usePersistence = archivesIndex
                .getClassByName(createDotName(persistenceFactoryClass)) != null;

        GeneratorContext context = buildContext(capabilities, config, appPaths, archivesIndex);

        ApplicationGenerator appGen = new ApplicationGenerator(config.packageName().orElse(DEFAULT_PACKAGE_NAME),
                new File(appPaths.getFirstProjectPath().toFile(), "target"))
                        .withDependencyInjection(new CDIDependencyInjectionAnnotator()).withPersistence(usePersistence)
                        .withGeneratorContext(context);
        List dependencies = new ArrayList<>();
        if (appModel != null) {
            for (ResolvedDependency i : appModel.getRuntimeDependencies()) {
                dependencies.add(i.getResolvedPaths().getSinglePath().toAbsolutePath().toString());
            }
        }

        addProcessGenerator(appPaths, usePersistence, appGen, dependencies, archivesIndex, onlyCodeAssets);

        if (context.getBuildContext().isDmnSupported()) {
            addDecisionGenerator(appPaths, appGen, false, dependencies);
        }

        return appGen;
    }

    private void addProcessGenerator(AppPaths appPaths, boolean usePersistence, ApplicationGenerator appGen,
            List dependencies, CompositeIndex index, boolean onlyCodeAssets)
            throws IOException {
        DotName workflowBuilderDotName = createDotName(workflowsBuilderClass);
        DotName workflowsAnnotationClassDotName = createDotName(workflowsAnnotationClass);
        List workflowBuilderClasses = index.getAnnotations(workflowsAnnotationClassDotName);
        List processes = new ArrayList();
        if (workflowBuilderClasses != null && !workflowBuilderClasses.isEmpty()) {
            for (AnnotationInstance workflowBuilderClass : workflowBuilderClasses) {
                ClassInfo clazz = workflowBuilderClass.target().asClass();

                String category = workflowBuilderClass.value("category") != null
                        ? workflowBuilderClass.value("category").asString()
                        : "";
                String categoryDescription = workflowBuilderClass.value("categoryDescription") != null
                        ? workflowBuilderClass.value("categoryDescription").asString()
                        : "";
                String resourcePathPrefix = workflowBuilderClass.value("resourcePathPrefix") != null
                        ? workflowBuilderClass.value("resourcePathPrefix").asString()
                        : "";

                String fqcn = clazz.name().toString();
                String sourceFolder = sourceFolder(appGen);

                String projects = appGen.context().getApplicationProperty("quarkus.automatiko.project-paths")
                        .orElse(null);
                List projectPaths = new ArrayList<>();
                projectPaths.addAll(Arrays.asList(appPaths.getProjectPaths()));
                if (projects != null) {
                    String[] extraProjectPaths = projects.split(",");
                    for (String extraPath : extraProjectPaths) {

                        for (Path projectPath : projectPaths.toArray(Path[]::new)) {
                            projectPaths.add(Paths.get(projectPath.toString(), extraPath).normalize());
                        }
                    }
                }
                Optional sourceFile = projectPaths.stream()
                        .map(projectPath -> projectPath + "/" + sourceFolder + "/" + fqcn.replace(".", "/") + ".java")
                        .filter(p -> Files.exists(Paths.get(p))).findFirst();

                if (sourceFile.isEmpty()) {
                    logger.warn("Unable to locate source file for fqcn {}", fqcn);
                    continue;
                }

                LambdaParser
                        .parseLambdas(sourceFile.get());

                try {
                    Class builderClass = Class.forName(fqcn, true, Thread.currentThread().getContextClassLoader());
                    Object workflowsBuilderInstance = builderClass
                            .getConstructor(new Class[0])
                            .newInstance();

                    List workflowsMethods = clazz.methods().stream()
                            .filter(m -> Modifier.isPublic(m.flags())
                                    && m.returnType().name().compareTo(workflowBuilderDotName) == 0)
                            .map(m -> m.name()).collect(Collectors.toList());

                    for (String methodName : workflowsMethods) {
                        Method method = workflowsBuilderInstance.getClass().getMethod(methodName);

                        WorkflowBuilder builder = (WorkflowBuilder) method.invoke(workflowsBuilderInstance);
                        ExecutableProcess process = builder.get();
                        process.setPackageName(builderClass.getPackageName() + ".generated");

                        // sets the category from annotation if not already set workflow
                        if (process.getMetaData("category") == null && !category.isBlank()) {
                            process.setMetaData("category", category);
                            process.setMetaData("categoryDescription", categoryDescription);
                        }

                        if (!resourcePathPrefix.isBlank()) {
                            process.setMetaData("referencePrefix", resourcePathPrefix);
                        }
                        processes.add(process);
                    }

                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    BuilderContext.clear();
                }
            }
        }
        ProcessCodegen generator;
        if (onlyCodeAssets) {
            generator = ProcessCodegen.ofProcesses(processes);
        } else {
            generator = appPaths.isJar ? ProcessCodegen.ofJar(processes, dependencies, appPaths.getJarPath())
                    : ProcessCodegen.ofPath(processes, dependencies, appPaths.getProjectPaths());
        }

        appGen.withGenerator(generator).withPersistence(usePersistence)
                .withClassLoader(Thread.currentThread().getContextClassLoader());
    }

    private void addDecisionGenerator(AppPaths appPaths, ApplicationGenerator appGen, boolean useMonitoring,
            List dependencies)
            throws IOException {
        DecisionCodegen generator = appPaths.isJar ? DecisionCodegen.ofJar(dependencies, appPaths.getJarPath())
                : DecisionCodegen.ofPath(dependencies, appPaths.getResourcePaths());

        appGen.withGenerator(generator).withMonitoring(useMonitoring);
    }

    private String toRuntimeSource(String sourceFolder, String className) {
        return sourceFolder + "/" + className.replace('.', '/') + ".java";
    }

    private String toClassName(String sourceName) {
        if (sourceName.startsWith("./")) {
            sourceName = sourceName.substring(2);
        }
        if (sourceName.endsWith(".java")) {
            sourceName = sourceName.substring(0, sourceName.length() - 5);
        } else if (sourceName.endsWith(".class")) {
            sourceName = sourceName.substring(0, sourceName.length() - 6);
        }
        return sourceName.replace('/', '.').replace('\\', '.');
    }

    private void writeGeneratedFile(GeneratedFile f, String sourceFolder, String location) throws IOException {
        if (location == null) {
            return;
        }

        String generatedClassFile = f.relativePath().replace(sourceFolder, "");

        Path sourceFilePath = pathOf(location, generatedClassFile);
        Path classFilePath = pathOf(location, generatedClassFile.replaceAll("\\.java", ".class"));

        Files.write(sourceFilePath, f.contents());

        AutomatikoBuildData.get().getGenerationContext().addClassToSourceMapping(classFilePath, sourceFilePath);
    }

    private Path pathOf(String location, String end) {
        Path path = Paths.get(location, end);
        path.getParent().toFile().mkdirs();
        return path;
    }

    private DotName createDotName(String name) {
        int lastDot = name.indexOf('.');
        if (lastDot < 0) {
            return DotName.createComponentized(null, name);
        }

        DotName lastDotName = null;
        while (lastDot > 0) {
            String local = name.substring(0, lastDot);
            name = name.substring(lastDot + 1);
            lastDot = name.indexOf('.');
            lastDotName = DotName.createComponentized(lastDotName, local);
        }

        int lastDollar = name.indexOf('$');
        if (lastDollar < 0) {
            return DotName.createComponentized(lastDotName, name);
        }
        DotName lastDollarName = null;
        while (lastDollar > 0) {
            String local = name.substring(0, lastDollar);
            name = name.substring(lastDollar + 1);
            lastDollar = name.indexOf('$');
            if (lastDollarName == null) {
                lastDollarName = DotName.createComponentized(lastDotName, local);
            } else {
                lastDollarName = DotName.createComponentized(lastDollarName, local, true);
            }
        }
        return DotName.createComponentized(lastDollarName, name, true);
    }

    private GeneratorContext buildContext(Capabilities capabilities, AutomatikoBuildTimeConfig config, AppPaths appPaths,
            IndexView index) {
        GeneratorContext generationContext = QuarkusGeneratorContext.ofResourcePath(appPaths.getResourceFiles()[0],
                appPaths.getFirstClassesPath().toFile());
        generationContext.withPackageName(config.packageName().orElse(DEFAULT_PACKAGE_NAME));

        generationContext
                .withBuildContext(new QuarkusApplicationBuildContext(config, className -> {
                    DotName classDotName = createDotName(className);
                    return !index.getAnnotations(classDotName).isEmpty() || index.getClassByName(classDotName) != null;

                }, className -> {
                    return index.getAllKnownImplementors(createDotName(className)).stream().map(c -> c.name().toString())
                            .collect(Collectors.toList());
                }, capability -> capabilities.isPresent(capability)));

        return AutomatikoBuildData.create(config, generationContext).getGenerationContext();
    }

    private static class AppPaths {

        private final Set projectPaths = new LinkedHashSet<>();
        private final List classesPaths = new ArrayList<>();

        private boolean isJar = false;

        private AppPaths(PathCollection paths) {
            for (Path path : paths) {
                PathType pathType = getPathType(path);
                switch (pathType) {
                    case CLASSES: {
                        classesPaths.add(path);
                        projectPaths.add(path.getParent().getParent());
                        break;
                    }
                    case TEST_CLASSES: {
                        projectPaths.add(path.getParent().getParent());
                        break;
                    }
                    case JAR: {
                        isJar = true;
                        classesPaths.add(path);
                        projectPaths.add(path.getParent().getParent());
                        break;
                    }
                    case UNKNOWN: {
                        classesPaths.add(path);
                        projectPaths.add(path);
                        break;
                    }
                }
            }
        }

        public Path getFirstProjectPath() {
            return projectPaths.iterator().next();
        }

        public Path getFirstClassesPath() {
            return classesPaths.get(0);
        }

        public Path[] getJarPath() {
            if (!isJar) {
                throw new IllegalStateException("Not a jar");
            }
            return classesPaths.toArray(new Path[classesPaths.size()]);
        }

        public File[] getResourceFiles() {
            return projectPaths.stream().map(p -> p.resolve("src/main/resources").toFile()).toArray(File[]::new);
        }

        public Path[] getResourcePaths() {
            return transformPaths(projectPaths, p -> p.resolve("src/main/resources"));
        }

        public Path[] getProjectPaths() {
            return transformPaths(projectPaths, Function.identity());
        }

        private Path[] transformPaths(Collection paths, Function f) {
            return paths.stream().map(f).toArray(Path[]::new);
        }

        private PathType getPathType(Path archiveLocation) {
            String path = archiveLocation.toString();
            if (path.endsWith("target" + File.separator + "classes")) {
                return PathType.CLASSES;
            }
            if (path.endsWith("target" + File.separator + "test-classes")) {
                return PathType.TEST_CLASSES;
            }
            if (path.endsWith(".jar") || path.endsWith(".jar.original")) {
                return PathType.JAR;
            }
            return PathType.UNKNOWN;
        }

        private enum PathType {
            CLASSES,
            TEST_CLASSES,
            JAR,
            UNKNOWN
        }
    }

    private static class SourceCode extends SimpleJavaFileObject {

        private String contents = null;

        public SourceCode(String className, String contents) throws Exception {
            super(new URI(className), Kind.SOURCE);
            this.contents = contents;
        }

        public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
            return contents;
        }

    }

    private String sourceFolder(ApplicationGenerator appGen) {
        String sourceFolder = appGen.context().getApplicationProperty("quarkus.automatiko.source-folder")
                .orElse("/src/main/java");
        if (sourceFolder.startsWith("/")) {
            sourceFolder = sourceFolder.substring(1);
        }

        return sourceFolder;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy