001/* 002 * Copyright 2017-2021 original authors 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * https://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package io.micronaut.maven.openapi; 017 018import io.micronaut.maven.AbstractMicronautMojo; 019import io.micronaut.openapi.generator.MicronautCodeGeneratorBuilder; 020import io.micronaut.openapi.generator.MicronautCodeGeneratorEntryPoint; 021import io.micronaut.openapi.generator.MicronautCodeGeneratorOptionsBuilder; 022import io.micronaut.openapi.generator.SerializationLibraryKind; 023import org.apache.maven.plugin.MojoExecutionException; 024import org.apache.maven.plugin.MojoFailureException; 025import org.apache.maven.plugins.annotations.Parameter; 026import org.apache.maven.project.MavenProject; 027 028import java.io.File; 029import java.util.List; 030import java.util.Locale; 031import java.util.Map; 032 033/** 034 * Base class for OpenAPI generator mojos. This provides the common 035 * parameters for all generators and the invoker logic. Subclasses 036 * must implement the {@link #isEnabled()} and {@link #configureBuilder(MicronautCodeGeneratorBuilder)} 037 * methods. 038 */ 039public abstract class AbstractOpenApiMojo extends AbstractMicronautMojo { 040 static final String MICRONAUT_OPENAPI_PREFIX = "micronaut.openapi"; 041 static final String IO_MICRONAUT_OPENAPI_PREFIX = "io.micronaut.openapi"; 042 043 /** 044 * The OpenAPI specification file path relative to the project's root path. 045 */ 046 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".definition", defaultValue = IO_MICRONAUT_OPENAPI_PREFIX + ".invoker", required = true) 047 protected File definitionFile; 048 049 /** 050 * The name of the package that can be used for various classes required for invocation. 051 */ 052 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".invoker.package.name", defaultValue = IO_MICRONAUT_OPENAPI_PREFIX + ".invoker", required = true) 053 protected String invokerPackageName; 054 055 /** 056 * The package name for the APIs (controller interfaces). 057 */ 058 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".api.package.name", defaultValue = IO_MICRONAUT_OPENAPI_PREFIX + ".api", required = true) 059 protected String apiPackageName; 060 061 /** 062 * The package name for the model classes. 063 */ 064 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".model.package.name", defaultValue = IO_MICRONAUT_OPENAPI_PREFIX + ".model", required = true) 065 protected String modelPackageName; 066 067 /** 068 * Whether to generate validation annotations for models and APIs. 069 */ 070 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".use.bean.validation", defaultValue = "true", required = true) 071 protected boolean useBeanValidation; 072 073 /** 074 * Flag to indicate whether to use the utils.OneOfImplementorAdditionalData related logic. 075 */ 076 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".use.one.of.interfaces", defaultValue = "true", required = true) 077 protected boolean useOneOfInterfaces; 078 079 /** 080 * Whether to use {@link java.util.Optional} for non-required model properties and API parameters. 081 */ 082 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".use.optional", defaultValue = "false", required = true) 083 protected boolean useOptional; 084 085 /** 086 * Whether to use reactor types for operation responses. 087 */ 088 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".use.reactive", defaultValue = "true", required = true) 089 protected boolean useReactive; 090 091 /** 092 * Configure the serialization library. 093 */ 094 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".serialization.framework", defaultValue = "MICRONAUT_SERDE_JACKSON", required = true) 095 protected String serializationFramework; 096 097 /** 098 * If true, the generated operation return types will be wrapped in HttpResponse. 099 */ 100 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".always.use.generate.http.response", defaultValue = "false", required = true) 101 protected boolean alwaysUseGenerateHttpResponse; 102 103 /** 104 * Wrap the operations response in HttpResponse object where non-200 HTTP status codes or additional headers are defined. 105 */ 106 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".generate.http.response.where.required", defaultValue = "true", required = true) 107 protected boolean generateHttpResponseWhereRequired; 108 109 /** 110 * If set to true, generated code will be fully compatible with KSP, but not 100% with KAPT. 111 */ 112 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".ksp", defaultValue = "false", required = true) 113 protected boolean ksp; 114 115 /** 116 * Configure the date-time format. 117 */ 118 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".date.time.format", defaultValue = "ZONED_DATETIME", required = true) 119 protected String dateTimeFormat; 120 121 /** 122 * Comma-separated values of output kinds to generate. The values are defined by the 123 * {@link MicronautCodeGeneratorEntryPoint.OutputKind} enum. 124 */ 125 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".outputs", required = true, defaultValue = "apis,models,supporting_files") 126 protected List<String> outputKinds; 127 128 /** 129 * The output directory to which all the sources will be generated. 130 */ 131 @Parameter(defaultValue = "${project.build.directory}/generated-sources/openapi", required = true) 132 protected File outputDirectory; 133 134 /** 135 * Define parameter mappings that allow using custom types for parameter binding. 136 * See {@link ParameterMapping} for details. 137 */ 138 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".parameterMappings") 139 protected List<ParameterMapping> parameterMappings; 140 141 /** 142 * Define parameter mappings that allow using custom types for parameter binding. 143 * See {@link ResponseBodyMapping} for details. 144 */ 145 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".responseBodyMappings") 146 protected List<ResponseBodyMapping> responseBodyMappings; 147 148 /** 149 * Add the schema mappings. 150 */ 151 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".schemaMapping") 152 protected Map<String, String> schemaMapping; 153 154 /** 155 * Add the import mappings. 156 */ 157 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".importMapping") 158 protected Map<String, String> importMapping; 159 160 /** 161 * Add the name mappings. 162 */ 163 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".nameMapping") 164 protected Map<String, String> nameMapping; 165 166 /** 167 * Add the type mappings. 168 */ 169 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".typeMapping") 170 protected Map<String, String> typeMapping; 171 172 /** 173 * Add the enum name mappings. 174 */ 175 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".enumNameMapping") 176 protected Map<String, String> enumNameMapping; 177 178 /** 179 * Add the model name mappings. 180 */ 181 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".modelNameMapping") 182 protected Map<String, String> modelNameMapping; 183 184 /** 185 * Add the inline schema name mappings. 186 */ 187 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".inlineSchemaNameMapping") 188 protected Map<String, String> inlineSchemaNameMapping; 189 190 /** 191 * Add the inline schema options. 192 */ 193 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".inlineSchemaOption") 194 protected Map<String, String> inlineSchemaOption; 195 196 /** 197 * Add the OpenAPI normalizer options. 198 */ 199 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".openapiNormalizer") 200 protected Map<String, String> openapiNormalizer; 201 202 /** 203 * Set the api name prefix. 204 */ 205 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".apiNamePrefix") 206 protected String apiNamePrefix; 207 208 /** 209 * Set the api name suffix. 210 */ 211 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".apiNameSuffix") 212 protected String apiNameSuffix; 213 214 /** 215 * Set the model name prefix. 216 */ 217 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".modelNamePrefix") 218 protected String modelNamePrefix; 219 220 /** 221 * Set the model name suffix. 222 */ 223 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".modelNameSuffix") 224 protected String modelNameSuffix; 225 226 /** 227 * Allows specifying the language of the generated code. 228 * 229 * @since 4.3.0 230 */ 231 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".lang", defaultValue = "java") 232 protected String lang; 233 234 /** 235 * If set to true, the generated enums check enum value with ignoring case. 236 */ 237 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".useEnumCaseInsensitive", defaultValue = "false") 238 protected boolean useEnumCaseInsensitive; 239 240 /** 241 * Additional annotations for enum type (class level annotations). 242 */ 243 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".additionalEnumTypeAnnotations") 244 protected List<String> additionalEnumTypeAnnotations; 245 246 /** 247 * Additional annotations for model type (class level annotations). 248 */ 249 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".additionalModelTypeAnnotations") 250 protected List<String> additionalModelTypeAnnotations; 251 252 /** 253 * Additional annotations for oneOf interfaces (class level annotations). 254 */ 255 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".additionalOneOfTypeAnnotations") 256 protected List<String> additionalOneOfTypeAnnotations; 257 258 /** 259 * Additional generator properties. 260 */ 261 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".additionalProperties") 262 protected Map<String, Object> additionalProperties; 263 264 /** 265 * If set to true, controller and client method will be generated with openAPI annotations. 266 */ 267 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".generateSwaggerAnnotations", defaultValue = "false") 268 protected boolean generateSwaggerAnnotations; 269 270 /** 271 * Set the implicit headers flag. 272 */ 273 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".implicitHeaders", defaultValue = "false") 274 protected boolean implicitHeaders; 275 276 /** 277 * Set the implicit headers regex. 278 */ 279 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".implicitHeadersRegex") 280 protected String implicitHeadersRegex; 281 282 /** 283 * Flag to indicate whether to use the "jakarta" or "javax" package. 284 */ 285 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".useJakartaEe", defaultValue = "true") 286 protected boolean useJakartaEe = true; 287 288 /** 289 * Sort method arguments to place required parameters before optional parameters. 290 * Default: true 291 */ 292 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".sortParamsByRequiredFlag", defaultValue = "true") 293 protected boolean sortParamsByRequiredFlag = true; 294 295 /** 296 * Skip examples defined in operations to avoid out of memory errors. 297 * Default: false 298 */ 299 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".skipOperationExample") 300 protected boolean skipOperationExample; 301 302 /** 303 * Skip sorting operations. 304 * Default: false 305 */ 306 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".skipSortingOperations") 307 protected boolean skipSortingOperations; 308 309 /** 310 * Character to use as a delimiter for the prefix. Default: '_' 311 */ 312 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".removeOperationIdPrefixDelimiter", defaultValue = "_") 313 protected String removeOperationIdPrefixDelimiter = "_"; 314 315 /** 316 * Count of delimiter for the prefix. Use -1 for last. Default: 1 317 */ 318 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".removeOperationIdPrefixCount", defaultValue = "1") 319 protected int removeOperationIdPrefixCount = 1; 320 321 /** 322 * Sort model properties to place required parameters before optional parameters. 323 */ 324 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".sortModelPropertiesByRequiredFlag", defaultValue = "true") 325 protected boolean sortModelPropertiesByRequiredFlag = true; 326 327 /** 328 * Whether to ensure parameter names are unique in an operation (rename parameters that are not). 329 */ 330 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".ensureUniqueParams", defaultValue = "true") 331 protected boolean ensureUniqueParams = true; 332 333 /** 334 * boolean, toggles whether Unicode identifiers are allowed in names or not, default is false. 335 */ 336 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".allowUnicodeIdentifiers") 337 protected boolean allowUnicodeIdentifiers; 338 339 /** 340 * Add form or body parameters to the beginning of the parameter list. 341 */ 342 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".prependFormOrBodyParameters") 343 protected boolean prependFormOrBodyParameters; 344 345 /** 346 * If set to true, generated code will be with suspend methods. Ony for kotlin generator. 347 * 348 * @since 4.8.0 349 */ 350 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".coroutines") 351 protected boolean coroutines; 352 353 /** 354 * Whether to generate sealed model interfaces and classes. Only for java generator. 355 * 356 * @since 4.8.0 357 */ 358 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".useSealed") 359 protected boolean useSealed; 360 361 /** 362 * If set to true, {@literal @}JsonInclude annotation will be with value ALWAYS for required properties in POJO's. 363 * 364 * @since 4.8.0 365 */ 366 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".jsonIncludeAlwaysForRequiredFields") 367 protected boolean jsonIncludeAlwaysForRequiredFields; 368 369 /** 370 * Generate or not required properties constructor. Only for java generator. 371 * 372 * @since 4.8.0 373 */ 374 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".requiredPropertiesInConstructor", defaultValue = "true") 375 protected boolean requiredPropertiesInConstructor = true; 376 377 /** 378 * If true, the generated controller interface will be without `@Controller` annotation. 379 * 380 * @since 4.8.0 381 */ 382 @Parameter(property = MICRONAUT_OPENAPI_PREFIX + ".generateControllerAsAbstract") 383 protected boolean generateControllerAsAbstract; 384 385 @Parameter(defaultValue = "${project}", readonly = true) 386 protected MavenProject project; 387 388 /** 389 * Determines if this mojo must be executed. 390 * 391 * @return true if the mojo is enabled 392 */ 393 protected abstract boolean isEnabled(); 394 395 /** 396 * Configures the OpenAPI generator. When this method is called, 397 * common properties shared by all generators have already been 398 * configured, so this method should only take care of configuring 399 * the generator specific parameters. 400 * 401 * @param builder the generator configuration builder 402 */ 403 protected abstract void configureBuilder(MicronautCodeGeneratorBuilder builder) throws MojoExecutionException; 404 405 @Override 406 public final void execute() throws MojoExecutionException, MojoFailureException { 407 if (!isEnabled()) { 408 getLog().debug(this.getClass().getSimpleName() + " is disabled"); 409 return; 410 } 411 412 project.addCompileSourceRoot(outputDirectory.getAbsolutePath()); 413 var builder = MicronautCodeGeneratorEntryPoint.builder() 414 .withDefinitionFile(definitionFile.toURI()) 415 .withOutputDirectory(outputDirectory) 416 .withOutputs( 417 outputKinds.stream().map(String::toUpperCase).map(MicronautCodeGeneratorEntryPoint.OutputKind::valueOf).toList().toArray(new MicronautCodeGeneratorEntryPoint.OutputKind[0]) 418 ) 419 .withOptions(options -> options 420 .withLang(MicronautCodeGeneratorOptionsBuilder.GeneratorLanguage.valueOf(lang.toUpperCase(Locale.ENGLISH))) 421 .withApiPackage(apiPackageName) 422 .withModelPackage(modelPackageName) 423 .withInvokerPackage(invokerPackageName) 424 .withBeanValidation(useBeanValidation) 425 .withUseOneOfInterfaces(useOneOfInterfaces) 426 .withOptional(useOptional) 427 .withReactive(useReactive) 428 .withSerializationLibrary(SerializationLibraryKind.valueOf(serializationFramework.toUpperCase(Locale.ENGLISH))) 429 .withGenerateHttpResponseAlways(alwaysUseGenerateHttpResponse) 430 .withGenerateHttpResponseWhereRequired(generateHttpResponseWhereRequired) 431 .withDateTimeFormat(MicronautCodeGeneratorOptionsBuilder.DateTimeFormat.valueOf(dateTimeFormat.toUpperCase(Locale.ENGLISH))) 432 .withParameterMappings(parameterMappings.stream() 433 .map(mapping -> new io.micronaut.openapi.generator.ParameterMapping( 434 mapping.getName(), 435 io.micronaut.openapi.generator.ParameterMapping.ParameterLocation.valueOf( 436 mapping.getLocation().name() 437 ), 438 mapping.getMappedType(), 439 mapping.getMappedName(), 440 mapping.isValidated() 441 )) 442 .toList() 443 ) 444 .withResponseBodyMappings(responseBodyMappings.stream() 445 .map(mapping -> new io.micronaut.openapi.generator.ResponseBodyMapping( 446 mapping.getHeaderName(), 447 mapping.getMappedBodyType(), 448 mapping.isListWrapper(), 449 mapping.isValidated() 450 )) 451 .toList() 452 ) 453 .withSchemaMapping(schemaMapping) 454 .withImportMapping(importMapping) 455 .withNameMapping(nameMapping) 456 .withTypeMapping(typeMapping) 457 .withEnumNameMapping(enumNameMapping) 458 .withModelNameMapping(modelNameMapping) 459 .withInlineSchemaNameMapping(inlineSchemaNameMapping) 460 .withInlineSchemaOption(inlineSchemaOption) 461 .withOpenapiNormalizer(openapiNormalizer) 462 .withApiNamePrefix(apiNamePrefix != null ? apiNamePrefix : "") 463 .withApiNameSuffix(apiNameSuffix != null ? apiNameSuffix : "") 464 .withModelNamePrefix(modelNamePrefix != null ? modelNamePrefix : "") 465 .withModelNameSuffix(modelNameSuffix != null ? modelNameSuffix : "") 466 .withGenerateSwaggerAnnotations(generateSwaggerAnnotations) 467 .withImplicitHeaders(implicitHeaders) 468 .withImplicitHeadersRegex(implicitHeadersRegex != null ? implicitHeadersRegex : "") 469 .withUseEnumCaseInsensitive(useEnumCaseInsensitive) 470 .withAdditionalEnumTypeAnnotations(additionalEnumTypeAnnotations) 471 .withAdditionalModelTypeAnnotations(additionalModelTypeAnnotations) 472 .withAdditionalOneOfTypeAnnotations(additionalOneOfTypeAnnotations) 473 .withAdditionalProperties(additionalProperties) 474 475 .withUseJakartaEe(useJakartaEe) 476 .withSortParamsByRequiredFlag(sortParamsByRequiredFlag) 477 .withSkipOperationExample(skipOperationExample) 478 .withSkipSortingOperations(skipSortingOperations) 479 .withRemoveOperationIdPrefixDelimiter(removeOperationIdPrefixDelimiter) 480 .withRemoveOperationIdPrefixCount(removeOperationIdPrefixCount) 481 .withSortModelPropertiesByRequiredFlag(sortModelPropertiesByRequiredFlag) 482 .withEnsureUniqueParams(ensureUniqueParams) 483 .withAllowUnicodeIdentifiers(allowUnicodeIdentifiers) 484 .withPrependFormOrBodyParameters(prependFormOrBodyParameters) 485 486 .withUseSealed(useSealed) 487 .withJsonIncludeAlwaysForRequiredFields(jsonIncludeAlwaysForRequiredFields) 488 .withRequiredPropertiesInConstructor(requiredPropertiesInConstructor) 489 .withGenerateControllerAsAbstract(generateControllerAsAbstract) 490 ); 491 configureBuilder(builder); 492 builder.build().generate(); 493 } 494}