 
                        
        
                        
        com.smartdoc.util.MojoUtils Maven / Gradle / Ivy
/*
 * smart-doc https://github.com/shalousun/smart-doc
 *
 * Copyright (C) 2018-2023 smart-doc
 *
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package com.smartdoc.util;
import com.google.gson.ExclusionStrategy;
import com.google.gson.FieldAttributes;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.power.common.util.FileUtil;
import com.power.common.util.StringUtil;
import com.power.doc.model.*;
import com.smartdoc.constant.GlobalConstants;
import org.apache.commons.lang3.StringUtils;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.project.*;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.*;
import static com.smartdoc.constant.GlobalConstants.FILE_SEPARATOR;
/**
 * @author xingzi 2019/12/07 21:19
 */
public class MojoUtils {
    /**
     * Gson Object
     */
    public final static Gson GSON = new GsonBuilder().addDeserializationExclusionStrategy(new ExclusionStrategy() {
        @Override
        public boolean shouldSkipField(FieldAttributes fieldAttributes) {
            return false;
        }
        @Override
        public boolean shouldSkipClass(Class> aClass) {
            return false;
        }
    }).create();
    /**
     * Build ApiConfig
     *
     * @param configFile       config file
     * @param projectName      project name
     * @param project          Maven project object
     * @param projectBuilder   ProjectBuilder
     * @param mavenSession     maven session
     * @param projectArtifacts project artifacts
     * @param log              maven plugin log
     * @return com.power.doc.model.ApiConfig
     * @throws MojoExecutionException MojoExecutionException
     */
    public static ApiConfig buildConfig(File configFile, String projectName, MavenProject project, ProjectBuilder projectBuilder
            , MavenSession mavenSession, List projectArtifacts, Log log) throws MojoExecutionException {
        try {
            ClassLoader classLoader = ClassLoaderUtil.getRuntimeClassLoader(project);
            String data = FileUtil.getFileContent(new FileInputStream(configFile));
            ApiConfig apiConfig = GSON.fromJson(data, ApiConfig.class);
            if (StringUtil.isEmpty(apiConfig.getCodePath())) {
                apiConfig.setCodePath(GlobalConstants.SOURCE_CODE_PATH);
            }
            apiConfig.setClassLoader(classLoader);
            List apiDataDictionaries = apiConfig.getDataDictionaries();
            List apiErrorCodes = apiConfig.getErrorCodeDictionaries();
            List apiConstants = apiConfig.getApiConstants();
            BodyAdvice responseBodyAdvice = apiConfig.getResponseBodyAdvice();
            BodyAdvice requestBodyAdvice = apiConfig.getRequestBodyAdvice();
            if (Objects.nonNull(apiErrorCodes)) {
                apiErrorCodes.forEach(
                        apiErrorCode -> {
                            String className = apiErrorCode.getEnumClassName();
                            apiErrorCode.setEnumClass(getClassByClassName(className, classLoader));
                        }
                );
            }
            if (Objects.nonNull(apiDataDictionaries)) {
                apiDataDictionaries.forEach(
                        apiDataDictionary -> {
                            String className = apiDataDictionary.getEnumClassName();
                            apiDataDictionary.setEnumClass(getClassByClassName(className, classLoader));
                        }
                );
            }
            if (Objects.nonNull(apiConstants)) {
                apiConstants.forEach(
                        apiConstant -> {
                            String className = apiConstant.getConstantsClassName();
                            apiConstant.setConstantsClass(getClassByClassName(className, classLoader));
                        }
                );
            }
            if (Objects.nonNull(responseBodyAdvice) && StringUtil.isNotEmpty(responseBodyAdvice.getClassName())) {
                responseBodyAdvice.setWrapperClass(getClassByClassName(responseBodyAdvice.getClassName(), classLoader));
            }
            if (Objects.nonNull(requestBodyAdvice) && StringUtil.isNotEmpty(requestBodyAdvice.getClassName())) {
                requestBodyAdvice.setWrapperClass(getClassByClassName(requestBodyAdvice.getClassName(), classLoader));
            }
            if (StringUtil.isEmpty(apiConfig.getProjectName()) && StringUtil.isEmpty(projectName)) {
                apiConfig.setProjectName(project.getName());
            } else if (StringUtil.isNotEmpty(apiConfig.getProjectName())
                    && "${project.artifactId}".equals(apiConfig.getProjectName())) {
                apiConfig.setProjectName(project.getArtifactId());
            } else if (StringUtil.isNotEmpty(projectName)) {
                apiConfig.setProjectName(projectName);
            }
            addSourcePaths(project, projectBuilder, mavenSession, apiConfig, projectArtifacts, log);
            return apiConfig;
        } catch (FileNotFoundException e) {
            e.printStackTrace();
            return null;
        }
    }
    /**
     * Obtain Class by class name
     *
     * @param className   class name
     * @param classLoader urls
     * @return Class
     */
    public static Class getClassByClassName(String className, ClassLoader classLoader) {
        if (StringUtils.isBlank(className)) {
            return null;
        }
        try {
            return classLoader.loadClass(className);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
            return null;
        }
    }
    /**
     * addSourcePath
     */
    private static void addSourcePaths(MavenProject project, ProjectBuilder projectBuilder, MavenSession mavenSession, ApiConfig apiConfig,
                                       List projectArtifacts, Log log) {
        List sourceCodePaths = new ArrayList<>();
        // key is module's artifact name, value is module's path
        Map modules = new HashMap<>(40);
        buildModules(project, projectBuilder, mavenSession, modules, apiConfig.getCodePath(), log);
        modules.forEach((key, modulePath) -> projectArtifacts.forEach(artifactName -> {
            if (artifactName.equals(key)) {
                sourceCodePaths.add(SourceCodePath.builder().setPath(modulePath));
            }
        }));
        sourceCodePaths.add(SourceCodePath.builder()
                .setPath(project.getBasedir() + FILE_SEPARATOR + apiConfig.getCodePath()));
        SourceCodePath[] codePaths = new SourceCodePath[sourceCodePaths.size()];
        sourceCodePaths.toArray(codePaths);
        List artifacts = new ArrayList<>();
        for (Map.Entry module : modules.entrySet()) {
            String artifactName = module.getKey().split(":")[1];
            artifacts.add(artifactName);
        }
        log.info("Artifacts that the current project depends on: " + GSON.toJson(artifacts));
        log.info("Smart-doc has loaded the source code path: " + GSON.toJson(sourceCodePaths)
                .replaceAll("\\\\", "/").replaceAll("//", "/"));
        apiConfig.setSourceCodePaths(codePaths);
    }
    /**
     * reference project to module
     *
     * @param project current maven project
     */
    private static void buildModules(MavenProject project, ProjectBuilder projectBuilder, MavenSession mavenSession, Map moduleList,
                                     String codePath, Log log) {
        Map referenceMavenProject = new HashMap<>(20);
        //if module's version is SNAPSHOT
        if (project.getProjectReferences().isEmpty()) {
            referenceMavenProject = collectProject(project, projectBuilder, mavenSession, log);
        }
        //if module's version isn't  SNAPSHOT
        else {
            addByProjectReference(referenceMavenProject, project.getProjectReferences());
        }
        for (Map.Entry mavenProject : referenceMavenProject.entrySet()) {
            if (log.isDebugEnabled()) {
                log.debug(project.getName() + " references mavenProject is: " + mavenProject.getValue().getName());
            }
            String artifactId = mavenProject.getValue().getModel().getArtifactId();
            String groupId = mavenProject.getValue().getGroupId();
            moduleList.put(groupId + ":" + artifactId, mavenProject.getValue().getBasedir() + FILE_SEPARATOR + codePath);
        }
    }
    /**
     * addByProjectReference
     *
     * @param referenceMavenProject target reference map
     * @param sourceProject         source reference map
     */
    public static void addByProjectReference(Map referenceMavenProject, Map sourceProject) {
        if (sourceProject.isEmpty()) {
            return;
        }
        referenceMavenProject.putAll(sourceProject);
        for (Map.Entry map : sourceProject.entrySet()) {
            addByProjectReference(referenceMavenProject, map.getValue().getProjectReferences());
        }
    }
    /**
     * load MavenProject from pom.xml
     *
     * @param project        current project
     * @param projectBuilder projectBuilder
     * @param session        maven session
     * @param log            log
     * @return Map
     */
    public static Map collectProject(MavenProject project, ProjectBuilder projectBuilder, MavenSession session, Log log) {
        Map mavenProjects = new HashMap<>(40);
        List pomPath = new ArrayList<>();
        getPomFilePath(getRootPath(project, log), pomPath);
        for (String s : pomPath) {
            File pomFile = new File(s);
            ProjectBuildingRequest request = new DefaultProjectBuildingRequest(session.getProjectBuildingRequest());
            try {
                MavenProject target = projectBuilder.build(pomFile, request).getProject();
                mavenProjects.put(target.getGroupId() + ":" + target.getArtifactId(), target);
            } catch (ProjectBuildingException e) {
                log.info(e.getMessage());
            }
        }
        return mavenProjects;
    }
    /**
     * @param file rootProject path
     * @param path all pom.xml path
     */
    private static void getPomFilePath(File file, List path) {
        File[] fs = file.listFiles();
        assert fs != null;
        for (File f : fs) {
            if (!f.isDirectory()) {
                if (f.getPath().endsWith("pom.xml")) {
                    path.add(f.getPath());
                }
            } else {
                getPomFilePath(f, path);
            }
        }
    }
    /**
     * RootParentPath
     *
     * @param project current Project
     */
    private static File getRootPath(MavenProject project, Log log) {
        if (project.hasParent()) {
            MavenProject mavenProject = project.getParent();
            if (log.isDebugEnabled()) {
                log.debug(project.getName() + " parent is: " + mavenProject.getName());
            }
            if (null != mavenProject) {
                if (mavenProject.getBasedir() == null) {
                    return project.getBasedir();
                } else {
                    return getRootPath(mavenProject, log);
                }
            } else {
                return project.getBasedir();
            }
        } else {
            return project.getBasedir();
        }
    }
}                   © 2015 - 2025 Weber Informatics LLC | Privacy Policy