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

org.openapitools.codegen.config.CodegenConfigurator Maven / Gradle / Ivy

There is a newer version: 3.3.4.2
Show newest version
/*
 * Copyright 2018 OpenAPI-Generator Contributors (https://openapi-generator.tech)
 * Copyright 2018 SmartBear Software
 *
 * Licensed 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 org.openapitools.codegen.config;

import com.fasterxml.jackson.annotation.JsonAnyGetter;
import com.fasterxml.jackson.annotation.JsonAnySetter;
import io.swagger.parser.OpenAPIParser;
import io.swagger.v3.core.util.Json;
import io.swagger.v3.oas.models.OpenAPI;
import io.swagger.v3.parser.core.models.AuthorizationValue;
import io.swagger.v3.parser.core.models.ParseOptions;
import io.swagger.v3.parser.core.models.SwaggerParseResult;
import org.apache.commons.lang3.Validate;
import org.openapitools.codegen.CliOption;
import org.openapitools.codegen.ClientOptInput;
import org.openapitools.codegen.ClientOpts;
import org.openapitools.codegen.CodegenConfig;
import org.openapitools.codegen.CodegenConfigLoader;
import org.openapitools.codegen.CodegenConstants;
import org.openapitools.codegen.SpecValidationException;
import org.openapitools.codegen.auth.AuthParser;
import org.openapitools.codegen.languages.CSharpNancyFXServerCodegen;
import org.openapitools.codegen.languages.CppQt5ClientCodegen;
import org.openapitools.codegen.languages.CppRestSdkClientCodegen;
import org.openapitools.codegen.languages.CppTizenClientCodegen;
import org.openapitools.codegen.languages.JavaJerseyServerCodegen;
import org.openapitools.codegen.languages.PhpLumenServerCodegen;
import org.openapitools.codegen.languages.PhpSlimServerCodegen;
import org.openapitools.codegen.languages.PhpZendExpressivePathHandlerServerCodegen;
import org.openapitools.codegen.languages.RubySinatraServerCodegen;
import org.openapitools.codegen.languages.ScalaAkkaClientCodegen;
import org.openapitools.codegen.languages.ScalaHttpClientCodegen;
import org.openapitools.codegen.languages.SwiftClientCodegen;
import org.openapitools.codegen.utils.ModelUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.io.IOException;
import java.io.Serializable;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;

import static org.apache.commons.lang3.StringUtils.isNotEmpty;

/**
 * A class that contains all codegen configuration properties a user would want to manipulate.
 * An instance could be created by deserializing a JSON file or being populated from CLI or Maven plugin parameters.
 * It also has a convenience method for creating a ClientOptInput class which is THE object DefaultGenerator.java needs
 * to generate code.
 */
public class CodegenConfigurator implements Serializable {

    public static final Logger LOGGER = LoggerFactory.getLogger(CodegenConfigurator.class);

    private static Map nameMigrationMap = new HashMap<>();
    static {
        nameMigrationMap.put("akka-scala", new ScalaAkkaClientCodegen().getName());
        nameMigrationMap.put("scala", new ScalaHttpClientCodegen().getName());
        nameMigrationMap.put("jaxrs", new JavaJerseyServerCodegen().getName());
        nameMigrationMap.put("qt5cpp", new CppQt5ClientCodegen().getName());
        nameMigrationMap.put("cpprest", new CppRestSdkClientCodegen().getName());
        nameMigrationMap.put("tizen", new CppTizenClientCodegen().getName());
        nameMigrationMap.put("sinatra", new RubySinatraServerCodegen().getName());
        nameMigrationMap.put("swift", new SwiftClientCodegen().getName());
        nameMigrationMap.put("lumen", new PhpLumenServerCodegen().getName());
        nameMigrationMap.put("slim", new PhpSlimServerCodegen().getName());
        nameMigrationMap.put("ze-ph", new PhpZendExpressivePathHandlerServerCodegen().getName());
        nameMigrationMap.put("nancyfx", new CSharpNancyFXServerCodegen().getName());
    }

    private String generatorName;
    private String inputSpec;
    private String outputDir;
    private boolean verbose;
    private boolean skipOverwrite;
    private boolean removeOperationIdPrefix;
    private boolean validateSpec;
    private boolean enablePostProcessFile;
    private String templateDir;
    private String auth;
    private String apiPackage;
    private String modelPackage;
    private String invokerPackage;
    private String modelNamePrefix;
    private String modelNameSuffix;
    private String groupId;
    private String artifactId;
    private String artifactVersion;
    private String library;
    private String ignoreFileOverride;
    private Map systemProperties = new HashMap();
    private Map instantiationTypes = new HashMap();
    private Map typeMappings = new HashMap();
    private Map additionalProperties = new HashMap();
    private Map importMappings = new HashMap();
    private Set languageSpecificPrimitives = new HashSet();
    private Map  reservedWordMappings = new HashMap();

    private String gitUserId="GIT_USER_ID";
    private String gitRepoId="GIT_REPO_ID";
    private String releaseNote="Minor update";
    private String httpUserAgent;

    private final Map dynamicProperties = new HashMap(); //the map that holds the JsonAnySetter/JsonAnyGetter values

    public CodegenConfigurator() {
        this.validateSpec = true;
        this.setOutputDir(".");
    }

    // TODO: When setLang is removed, please remove nameMigrationMap and its usage(s).
    /**
     * Set the "language". This has drifted away from language-only to include framework and hyphenated generator types as well as language.
     * 

* NOTE: This will eventually become language only again. It is deprecated in its current state. *

* * @deprecated Please use {@link #setGeneratorName(String)}, as generators are no longer identified only by language. We may reuse language in the future. * @param lang The generator name. Previously, language name only. * @return The fluent instance of {@link CodegenConfigurator} */ @Deprecated public CodegenConfigurator setLang(String lang) { this.setGeneratorName(lang); return this; } /** * Sets the name of the target generator. * * The generator's name is used to uniquely identify the generator as a mechanism to lookup the desired implementation * at runtime. * * @param generatorName The name of the generator. * @return The fluent instance of {@link CodegenConfigurator} */ public CodegenConfigurator setGeneratorName(final String generatorName) { if (nameMigrationMap.containsKey(generatorName)) { String newValue = nameMigrationMap.get(generatorName); LOGGER.warn(String.format(Locale.ROOT, "The name '%s' is a deprecated. Please update to the new name of '%s'.", generatorName, newValue)); this.generatorName = newValue; } else { this.generatorName = generatorName; } return this; } public CodegenConfigurator setInputSpec(String inputSpec) { this.inputSpec = inputSpec; return this; } public String getInputSpec() { return inputSpec; } public String getOutputDir() { return outputDir; } public CodegenConfigurator setOutputDir(String outputDir) { this.outputDir = toAbsolutePathStr(outputDir); return this; } public String getModelPackage() { return modelPackage; } public CodegenConfigurator setModelPackage(String modelPackage) { this.modelPackage = modelPackage; return this; } public String getModelNamePrefix() { return modelNamePrefix; } public CodegenConfigurator setModelNamePrefix(String prefix) { this.modelNamePrefix = prefix; return this; } public boolean getRemoveOperationIdPrefix() { return removeOperationIdPrefix; } public CodegenConfigurator setRemoveOperationIdPrefix(boolean removeOperationIdPrefix) { this.removeOperationIdPrefix = removeOperationIdPrefix; return this; } public boolean getEnablePostProcessFile() { return enablePostProcessFile; } public CodegenConfigurator setEnablePostProcessFile(boolean enablePostProcessFile) { this.enablePostProcessFile = enablePostProcessFile; return this; } public String getModelNameSuffix() { return modelNameSuffix; } public CodegenConfigurator setModelNameSuffix(String suffix) { this.modelNameSuffix = suffix; return this; } public boolean isVerbose() { return verbose; } public CodegenConfigurator setVerbose(boolean verbose) { this.verbose = verbose; return this; } public boolean isValidateSpec() { return validateSpec; } public CodegenConfigurator setValidateSpec(final boolean validateSpec) { this.validateSpec = validateSpec; return this; } public boolean isSkipOverwrite() { return skipOverwrite; } public CodegenConfigurator setSkipOverwrite(boolean skipOverwrite) { this.skipOverwrite = skipOverwrite; return this; } /** * Gets the "language". This has drifted away from language-only to include framework and hyphenated generator types as well as language. *

* NOTE: This will eventually become language only again. It is deprecated in its current state. *

* * @deprecated Please use {@link #getGeneratorName()}, as generators are no longer identified only by language. We may reuse language in the future. * * @return A string which defines the generator. */ @Deprecated public String getLang() { return getGeneratorName(); } public String getGeneratorName() { return generatorName; } public String getTemplateDir() { return templateDir; } public CodegenConfigurator setTemplateDir(String templateDir) { File f = new File(templateDir); // check to see if the folder exists if (!(f.exists() && f.isDirectory())) { throw new IllegalArgumentException("Template directory " + templateDir + " does not exist."); } this.templateDir = f.getAbsolutePath(); return this; } public String getAuth() { return auth; } public CodegenConfigurator setAuth(String auth) { this.auth = auth; return this; } public String getApiPackage() { return apiPackage; } public CodegenConfigurator setApiPackage(String apiPackage) { this.apiPackage = apiPackage; return this; } public String getInvokerPackage() { return invokerPackage; } public CodegenConfigurator setInvokerPackage(String invokerPackage) { this.invokerPackage = invokerPackage; return this; } public String getGroupId() { return groupId; } public CodegenConfigurator setGroupId(String groupId) { this.groupId = groupId; return this; } public String getArtifactId() { return artifactId; } public CodegenConfigurator setArtifactId(String artifactId) { this.artifactId = artifactId; return this; } public String getArtifactVersion() { return artifactVersion; } public CodegenConfigurator setArtifactVersion(String artifactVersion) { this.artifactVersion = artifactVersion; return this; } public Map getSystemProperties() { return systemProperties; } public CodegenConfigurator setSystemProperties(Map systemProperties) { this.systemProperties = systemProperties; return this; } public CodegenConfigurator addSystemProperty(String key, String value) { this.systemProperties.put(key, value); return this; } public Map getInstantiationTypes() { return instantiationTypes; } public CodegenConfigurator setInstantiationTypes(Map instantiationTypes) { this.instantiationTypes = instantiationTypes; return this; } public CodegenConfigurator addInstantiationType(String key, String value) { this.instantiationTypes.put(key, value); return this; } public Map getTypeMappings() { return typeMappings; } public CodegenConfigurator setTypeMappings(Map typeMappings) { this.typeMappings = typeMappings; return this; } public CodegenConfigurator addTypeMapping(String key, String value) { this.typeMappings.put(key, value); return this; } public Map getAdditionalProperties() { return additionalProperties; } public CodegenConfigurator setAdditionalProperties(Map additionalProperties) { this.additionalProperties = additionalProperties; return this; } public CodegenConfigurator addAdditionalProperty(String key, Object value) { this.additionalProperties.put(key, value); return this; } public Map getImportMappings() { return importMappings; } public CodegenConfigurator setImportMappings(Map importMappings) { this.importMappings = importMappings; return this; } public CodegenConfigurator addImportMapping(String key, String value) { this.importMappings.put(key, value); return this; } public Set getLanguageSpecificPrimitives() { return languageSpecificPrimitives; } public CodegenConfigurator setLanguageSpecificPrimitives(Set languageSpecificPrimitives) { this.languageSpecificPrimitives = languageSpecificPrimitives; return this; } public CodegenConfigurator addLanguageSpecificPrimitive(String value) { this.languageSpecificPrimitives.add(value); return this; } public String getLibrary() { return library; } public CodegenConfigurator setLibrary(String library) { this.library = library; return this; } public String getGitUserId() { return gitUserId; } public CodegenConfigurator setGitUserId(String gitUserId) { this.gitUserId = gitUserId; return this; } public String getGitRepoId() { return gitRepoId; } public CodegenConfigurator setGitRepoId(String gitRepoId) { this.gitRepoId = gitRepoId; return this; } public String getReleaseNote() { return releaseNote; } public CodegenConfigurator setReleaseNote(String releaseNote) { this.releaseNote = releaseNote; return this; } public String getHttpUserAgent() { return httpUserAgent; } public CodegenConfigurator setHttpUserAgent(String httpUserAgent) { this.httpUserAgent= httpUserAgent; return this; } public Map getReservedWordsMappings() { return reservedWordMappings; } public CodegenConfigurator setReservedWordsMappings(Map reservedWordsMappings) { this.reservedWordMappings = reservedWordsMappings; return this; } public CodegenConfigurator addAdditionalReservedWordMapping(String key, String value) { this.reservedWordMappings.put(key, value); return this; } public String getIgnoreFileOverride() { return ignoreFileOverride; } public CodegenConfigurator setIgnoreFileOverride(final String ignoreFileOverride) { this.ignoreFileOverride = ignoreFileOverride; return this; } public ClientOptInput toClientOptInput() { Validate.notEmpty(generatorName, "language/generatorName must be specified"); Validate.notEmpty(inputSpec, "input spec must be specified"); setVerboseFlags(); setSystemProperties(); CodegenConfig config = CodegenConfigLoader.forName(generatorName); config.setInputSpec(inputSpec); config.setOutputDir(outputDir); config.setSkipOverwrite(skipOverwrite); config.setIgnoreFilePathOverride(ignoreFileOverride); config.setRemoveOperationIdPrefix(removeOperationIdPrefix); config.setEnablePostProcessFile(enablePostProcessFile); config.instantiationTypes().putAll(instantiationTypes); config.typeMapping().putAll(typeMappings); config.importMapping().putAll(importMappings); config.languageSpecificPrimitives().addAll(languageSpecificPrimitives); config.reservedWordsMappings().putAll(reservedWordMappings); checkAndSetAdditionalProperty(apiPackage, CodegenConstants.API_PACKAGE); checkAndSetAdditionalProperty(modelPackage, CodegenConstants.MODEL_PACKAGE); checkAndSetAdditionalProperty(invokerPackage, CodegenConstants.INVOKER_PACKAGE); checkAndSetAdditionalProperty(groupId, CodegenConstants.GROUP_ID); checkAndSetAdditionalProperty(artifactId, CodegenConstants.ARTIFACT_ID); checkAndSetAdditionalProperty(artifactVersion, CodegenConstants.ARTIFACT_VERSION); checkAndSetAdditionalProperty(templateDir, toAbsolutePathStr(templateDir), CodegenConstants.TEMPLATE_DIR); checkAndSetAdditionalProperty(modelNamePrefix, CodegenConstants.MODEL_NAME_PREFIX); checkAndSetAdditionalProperty(modelNameSuffix, CodegenConstants.MODEL_NAME_SUFFIX); checkAndSetAdditionalProperty(gitUserId, CodegenConstants.GIT_USER_ID); checkAndSetAdditionalProperty(gitRepoId, CodegenConstants.GIT_REPO_ID); checkAndSetAdditionalProperty(releaseNote, CodegenConstants.RELEASE_NOTE); checkAndSetAdditionalProperty(httpUserAgent, CodegenConstants.HTTP_USER_AGENT); handleDynamicProperties(config); if (isNotEmpty(library)) { config.setLibrary(library); } config.additionalProperties().putAll(additionalProperties); ClientOptInput input = new ClientOptInput() .config(config); final List authorizationValues = AuthParser.parse(auth); ParseOptions options = new ParseOptions(); options.setResolve(true); SwaggerParseResult result = new OpenAPIParser().readLocation(inputSpec, authorizationValues, options); Set validationMessages = new HashSet<>(result.getMessages()); OpenAPI specification = result.getOpenAPI(); // NOTE: We will only expose errors+warnings if there are already errors in the spec. if (validationMessages.size() > 0) { Set warnings = new HashSet<>(); if (specification != null) { List unusedModels = ModelUtils.getUnusedSchemas(specification); if (unusedModels != null) unusedModels.forEach(name -> warnings.add("Unused model: " + name)); } if (this.isValidateSpec()) { StringBuilder sb = new StringBuilder(); sb.append("There were issues with the specification. The option can be disabled via validateSpec (Maven/Gradle) or --skip-validate-spec (CLI)."); sb.append(System.lineSeparator()); SpecValidationException ex = new SpecValidationException(sb.toString()); ex.setErrors(validationMessages); ex.setWarnings(warnings); throw ex; } else { StringBuilder sb = new StringBuilder(); sb.append("There were issues with the specification, but validation has been explicitly disabled."); sb.append(System.lineSeparator()); sb.append("Errors: ").append(System.lineSeparator()); validationMessages.forEach(msg -> sb.append("\t-").append(msg).append(System.lineSeparator()) ); if (!warnings.isEmpty()) { sb.append("Warnings: ").append(System.lineSeparator()); warnings.forEach(msg -> sb.append("\t-").append(msg).append(System.lineSeparator()) ); } LOGGER.warn(sb.toString()); } } input.opts(new ClientOpts()) .openAPI(specification); return input; } @JsonAnySetter public CodegenConfigurator addDynamicProperty(String name, Object value) { dynamicProperties.put(name, value); return this; } @JsonAnyGetter public Map getDynamicProperties() { return dynamicProperties; } private void handleDynamicProperties(CodegenConfig codegenConfig) { for (CliOption langCliOption : codegenConfig.cliOptions()) { String opt = langCliOption.getOpt(); if (dynamicProperties.containsKey(opt)) { codegenConfig.additionalProperties().put(opt, dynamicProperties.get(opt)); } else if(systemProperties.containsKey(opt)) { codegenConfig.additionalProperties().put(opt, systemProperties.get(opt)); } } } private void setVerboseFlags() { if (!verbose) { return; } LOGGER.info("\nVERBOSE MODE: ON. Additional debug options are injected" + "\n - [debugOpenAPI] prints the OpenAPI specification as interpreted by the codegen" + "\n - [debugModels] prints models passed to the template engine" + "\n - [debugOperations] prints operations passed to the template engine" + "\n - [debugSupportingFiles] prints additional data passed to the template engine"); System.setProperty("debugOpenAPI", ""); System.setProperty("debugModels", ""); System.setProperty("debugOperations", ""); System.setProperty("debugSupportingFiles", ""); } private void setSystemProperties() { for (Map.Entry entry : systemProperties.entrySet()) { System.setProperty(entry.getKey(), entry.getValue()); } } private static String toAbsolutePathStr(String path) { if (isNotEmpty(path)) { return Paths.get(path).toAbsolutePath().toString(); } return path; } private void checkAndSetAdditionalProperty(String property, String propertyKey) { checkAndSetAdditionalProperty(property, property, propertyKey); } private void checkAndSetAdditionalProperty(String property, String valueToSet, String propertyKey) { if (isNotEmpty(property)) { additionalProperties.put(propertyKey, valueToSet); } } public static CodegenConfigurator fromFile(String configFile) { if (isNotEmpty(configFile)) { try { return Json.mapper().readValue(new File(configFile), CodegenConfigurator.class); } catch (IOException e) { LOGGER.error("Unable to deserialize config file: " + configFile, e); } } return null; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy