diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index f17983f9ce..47129f073c 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -18,6 +18,7 @@ swagger-parser-v3 = "2.1.22" javaparser = "3.26.2" commons-codec = "1.17.1" guava = "33.3.0-jre" +spring-boot = "3.3.3" micronaut = "4.6.5" micronaut-platform = "4.6.1" @@ -36,6 +37,7 @@ micronaut-kotlin = "4.4.0" micronaut-logging = "1.4.0" micronaut-session = "4.4.0" micronaut-grpc = "4.7.0" +micronaut-spring = "5.8.0" micronaut-docs = "2.0.0" [libraries] @@ -68,6 +70,7 @@ micronaut-data = { module = "io.micronaut.data:micronaut-data-bom", version.ref micronaut-test = { module = "io.micronaut.test:micronaut-test-bom", version.ref = "micronaut-test" } micronaut-kotlin = { module = "io.micronaut.kotlin:micronaut-kotlin-bom", version.ref = "micronaut-kotlin" } micronaut-grpc = { module = "io.micronaut.grpc:micronaut-grpc-bom", version.ref = "micronaut-grpc" } +micronaut-spring = { module = "io.micronaut.spring:micronaut-spring-bom", version.ref = "micronaut-spring" } micronaut-platform = { module = "io.micronaut.platform:micronaut-platform", version.ref = "micronaut-platform"} micronaut-gradle-plugin = { module = "io.micronaut.gradle:micronaut-minimal-plugin", version.ref = "micronaut-gradle-plugin"} @@ -79,6 +82,7 @@ android-annotation = { module = "androidx.annotation:annotation", version.ref = javaparser = { module = "com.github.javaparser:javaparser-symbol-solver-core", version.ref = "javaparser" } commons-codec = { module = "commons-codec:commons-codec", version.ref = "commons-codec" } guava = { module = "com.google.guava:guava", version.ref = "guava" } +spring-boot-dependencies = { module = "org.springframework.boot:spring-boot-dependencies", version.ref = "spring-boot" } openapi-generator = { module = "org.openapitools:openapi-generator", version.ref = "openapi-generator" } swagger-parser = { module = "io.swagger:swagger-parser", version.ref = "swagger-parser" } diff --git a/openapi/src/main/java/io/micronaut/openapi/visitor/ElementUtils.java b/openapi/src/main/java/io/micronaut/openapi/visitor/ElementUtils.java index 0ef8100482..f679620eb2 100644 --- a/openapi/src/main/java/io/micronaut/openapi/visitor/ElementUtils.java +++ b/openapi/src/main/java/io/micronaut/openapi/visitor/ElementUtils.java @@ -27,6 +27,7 @@ import io.micronaut.http.HttpRequest; import io.micronaut.http.HttpResponse; import io.micronaut.http.annotation.Header; +import io.micronaut.http.annotation.RequestAttribute; import io.micronaut.http.multipart.FileUpload; import io.micronaut.inject.annotation.AnnotationMetadataHierarchy; import io.micronaut.inject.ast.ClassElement; @@ -246,8 +247,10 @@ public static boolean isIgnoredParameter(TypedElement parameter) { || parameter.isAnnotationPresent(Hidden.class) || parameter.isAnnotationPresent(JsonIgnore.class) || parameter.isAnnotationPresent(Header.class) && parameter.getType().isAssignable(Map.class) + || parameter.isAnnotationPresent(RequestAttribute.class) || parameter.booleanValue(Parameter.class, PROP_HIDDEN).orElse(false) || parameter.hasAnnotation("io.micronaut.session.annotation.SessionValue") + || parameter.hasAnnotation("org.springframework.web.bind.annotation.RequestAttribute") || parameter.hasAnnotation("org.springframework.web.bind.annotation.SessionAttribute") || parameter.hasAnnotation("org.springframework.web.bind.annotation.SessionAttributes") || parameter.hasAnnotation("jakarta.ws.rs.core.Context") diff --git a/openapi/src/main/java/io/micronaut/openapi/visitor/SchemaDefinitionUtils.java b/openapi/src/main/java/io/micronaut/openapi/visitor/SchemaDefinitionUtils.java index e3b548d820..0f6382376e 100644 --- a/openapi/src/main/java/io/micronaut/openapi/visitor/SchemaDefinitionUtils.java +++ b/openapi/src/main/java/io/micronaut/openapi/visitor/SchemaDefinitionUtils.java @@ -2479,10 +2479,10 @@ private static void processPropertyElements(OpenAPI openAPI, VisitorContext cont if (publicField instanceof MemberElement memberEl && (memberEl.getDeclaringType().getType().getName().equals(type.getName()) || isGetterOverridden)) { - ClassElement fieldType = publicField.getGenericType(); if (withJsonView && !allowedByJsonView(publicField, classLvlJsonViewClasses, jsonViewClass, context)) { continue; } + ClassElement fieldType = publicField.getGenericType(); Schema propertySchema = resolveSchema(openAPI, publicField, fieldType, context, mediaTypes, jsonViewClass, fieldJavadoc, classJavadoc); diff --git a/openapi/src/test/groovy/io/micronaut/openapi/visitor/OpenApiControllerVisitorSpec.groovy b/openapi/src/test/groovy/io/micronaut/openapi/visitor/OpenApiControllerVisitorSpec.groovy index b02ed99369..10d0d4ed3c 100644 --- a/openapi/src/test/groovy/io/micronaut/openapi/visitor/OpenApiControllerVisitorSpec.groovy +++ b/openapi/src/test/groovy/io/micronaut/openapi/visitor/OpenApiControllerVisitorSpec.groovy @@ -2436,6 +2436,76 @@ class TestController { } } +@Singleton +class MyBean {} +''') + when: "The OpenAPI is retrieved" + OpenAPI openAPI = Utils.testReference + + then: "the state is correct" + openAPI != null + + when: + Operation operation = openAPI.paths.get("/test").post + + then: + operation + + and: + operation.requestBody + operation.requestBody.content + operation.requestBody.content.size() == 1 + operation.requestBody.content."application/json" + operation.requestBody.content."application/json".schema + operation.requestBody.content."application/json".schema.$ref == "#/components/schemas/SimpleBody" + } + + void "test locale property"() { + given: + buildBeanDefinition('test.MyBean', ''' +package test; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import io.micronaut.http.HttpResponse; +import io.micronaut.http.annotation.Controller; +import io.micronaut.http.annotation.Get; +import io.swagger.v3.oas.annotations.Hidden; +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.links.Link; +import io.swagger.v3.oas.annotations.links.LinkParameter; +import io.swagger.v3.oas.annotations.responses.ApiResponse; +import jakarta.inject.Singleton; + +import java.util.List; +import java.util.Locale; + +@Controller("/test") +class TestController { + + @Operation(summary = "Request avatar info", operationId = "requestAvatar", responses = { + @ApiResponse(responseCode = "200", description = "OKI", links = { + @Link(name = "Download Avatar", operationId = "findAllWorkflowState", parameters = { + @LinkParameter(name = "workflowId", expression = "$response.body#/id"), + }) + }) + }) + @Get + public HttpResponse>> endpoint() { + return null; + } + +} + +class ResponseObject { + + public T body; +} + +class Dto { + + public Locale locale; +} + @Singleton class MyBean {} ''') diff --git a/settings.gradle b/settings.gradle index f28f1f40db..3a38bd9ac9 100644 --- a/settings.gradle +++ b/settings.gradle @@ -8,6 +8,7 @@ pluginManagement { plugins { id 'io.micronaut.build.shared.settings' version '7.2.1' + id "dev.aga.gradle.version-catalog-generator" version "1.5.0" } enableFeaturePreview 'TYPESAFE_PROJECT_ACCESSORS' @@ -24,6 +25,7 @@ include 'docs-examples:example-java' include 'docs-examples:example-kotlin' include 'test-suite-java-client-generator' include 'test-suite-java-jaxrs' +include 'test-suite-java-spring' include 'test-suite-java-server-generator' include 'test-suite-kotlin-kapt-client-generator' include 'test-suite-kotlin-kapt-server-generator' @@ -35,6 +37,9 @@ dependencyResolutionManagement { repositories { mavenCentral() } + versionCatalogs { + generator.generate("spring") { from(toml("spring-boot-dependencies")) } + } } micronautBuild { @@ -52,4 +57,5 @@ micronautBuild { importMicronautCatalog("micronaut-session") importMicronautCatalog("micronaut-jaxrs") importMicronautCatalog("micronaut-grpc") + importMicronautCatalog("micronaut-spring") } diff --git a/src/main/docs/guide/spring.adoc b/src/main/docs/guide/spring.adoc new file mode 100644 index 0000000000..aeb0b82c12 --- /dev/null +++ b/src/main/docs/guide/spring.adoc @@ -0,0 +1 @@ +You can use `micronaut-openapi` to build openapi specification for `Spring` / `Spring Boot` applications. In this case, you do not need to change anything in the code. You can continue to use spring, and use micronaut as a replacement for such libraries as `springdoc-openapi`, `springfox`, `swagger` etc. diff --git a/src/main/docs/guide/spring/springWithGradle.adoc b/src/main/docs/guide/spring/springWithGradle.adoc new file mode 100644 index 0000000000..9a917da5a1 --- /dev/null +++ b/src/main/docs/guide/spring/springWithGradle.adoc @@ -0,0 +1,25 @@ +To use micronaut-openapi with spring in gradle add this code to you `build.gradle`: + +.build.gradle +[source,groovy] +---- +dependencies { + + // add to annotationProcessor and compileOnly blocks next libraries: + + annotationProcessor platform("io.micronaut.platform:micronaut-platform:$micronautVersion") + annotationProcessor "io.micronaut:micronaut-inject-java" + annotationProcessor "io.micronaut.spring:micronaut-spring-annotation" + annotationProcessor "io.micronaut.spring:micronaut-spring-web-annotation" + annotationProcessor "io.micronaut.spring:micronaut-spring-boot-annotation" + annotationProcessor "io.micronaut.openapi:micronaut-openapi" + + compileOnly platform("io.micronaut.platform:micronaut-platform:$micronautVersion") + compileOnly "io.micronaut:micronaut-inject-java" + compileOnly "io.micronaut.openapi:micronaut-openapi-annotations" + compileOnly "io.micronaut.serde:micronaut-serde-api" + +} +---- + +For kotlin just change block `annotationProcessor` to `kapt` or `ksp`. diff --git a/src/main/docs/guide/spring/springWithMaven.adoc b/src/main/docs/guide/spring/springWithMaven.adoc new file mode 100644 index 0000000000..4899a369d7 --- /dev/null +++ b/src/main/docs/guide/spring/springWithMaven.adoc @@ -0,0 +1,131 @@ +To use micronaut-openapi with spring in maven add this code to your `pom.xml` + +.pom.xml +[source,xml] +---- + + + io.micronaut + micronaut-inject-java + ${micronaut.core.version} + provided + + + io.micronaut.openapi + micronaut-openapi-annotations + ${micronaut.openapi.version} + provided + + + io.micronaut.serde + micronaut-serde-api + ${micronaut.serde.version} + provided + + + + + + + org.apache.maven.plugins + maven-compiler-plugin + + + + io.micronaut + micronaut-inject-java + ${micronaut.core.version} + + + io.micronaut.spring + micronaut-spring-annotation + ${micronaut.spring.version} + + + io.micronaut.spring + micronaut-spring-web-annotation + ${micronaut.spring.version} + + + io.micronaut.spring + micronaut-spring-boot-annotation + ${micronaut.spring.version} + + + io.micronaut.openapi + micronaut-openapi + ${micronaut.openapi.version} + + + + + + +---- + +For kotlin +.pom.xml +[source,xml] +---- + + + io.micronaut + micronaut-inject-java + ${micronaut.core.version} + provided + + + io.micronaut.openapi + micronaut-openapi-annotations + ${micronaut.openapi.version} + provided + + + io.micronaut.serde + micronaut-serde-api + ${micronaut.serde.version} + provided + + + + + + kapt + + kapt + + + + ${project.basedir}/src/main/kotlin + + + + io.micronaut + micronaut-inject-java + ${micronaut.core.version} + + + io.micronaut.spring + micronaut-spring-annotation + ${micronaut.spring.version} + + + io.micronaut.spring + micronaut-spring-web-annotation + ${micronaut.spring.version} + + + io.micronaut.spring + micronaut-spring-boot-annotation + ${micronaut.spring.version} + + + io.micronaut.openapi + micronaut-openapi + ${micronaut.openapi.version} + + + + + +---- diff --git a/src/main/docs/guide/spring/springWithOpenApiView.adoc b/src/main/docs/guide/spring/springWithOpenApiView.adoc new file mode 100644 index 0000000000..55c690f718 --- /dev/null +++ b/src/main/docs/guide/spring/springWithOpenApiView.adoc @@ -0,0 +1,26 @@ +To use micronaut openapi views (Swagger UI, OpenAPi Explorer, Redoc, RapiDoc) you need to add static resources to Spring configuration like this: + +.WebConfig.java +[source,java] +---- +package org.my.company; + +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.EnableWebMvc; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +@Configuration +@EnableWebMvc +public class WebConfig extends WebMvcConfigurer { + + @Override + public void addResourceHandlers(ResourceHandlerRegistry registry) { + + registry.addResourceHandler("/swagger-ui/**") + .addResourceLocations("classpath:/META-INF/swagger/views/swagger-ui/"); + registry.addResourceHandler("/swagger/**") + .addResourceLocations("classpath:/META-INF/swagger/"); + } +} +---- diff --git a/src/main/docs/guide/toc.yml b/src/main/docs/guide/toc.yml index d405a0be54..73d7e28dae 100644 --- a/src/main/docs/guide/toc.yml +++ b/src/main/docs/guide/toc.yml @@ -30,6 +30,10 @@ endpoints: endpointservers: Endpoints Servers endpointssecurityrequirements: Endpoints Security Requirements endpointspath: Endpoints Path +spring: + title: Micronaut OpenAPI with Spring + springWithGradle: Spring with Gradle + springWithMaven: Spring with Maven micronautOpenApiAnnotations: title: Micronaut OpenAPI annotations openapidecorator: '@OpenAPIDecorator' diff --git a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/Application.java b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/Application.java similarity index 91% rename from test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/Application.java rename to test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/Application.java index f828f687c0..9982a69c2c 100644 --- a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/Application.java +++ b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/Application.java @@ -1,4 +1,4 @@ -package io.micronaut.open.jaxrs; +package io.micronaut.openapi.jaxrs; import io.micronaut.runtime.Micronaut; import io.swagger.v3.oas.annotations.OpenAPIDefinition; diff --git a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiExposedTest.java b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiExposedTest.java similarity index 94% rename from test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiExposedTest.java rename to test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiExposedTest.java index c840566982..819d29d98b 100644 --- a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiExposedTest.java +++ b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiExposedTest.java @@ -1,4 +1,4 @@ -package io.micronaut.open.jaxrs; +package io.micronaut.openapi.jaxrs; import io.micronaut.http.client.BlockingHttpClient; import io.micronaut.http.client.HttpClient; diff --git a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiGeneratedTest.java b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiGeneratedTest.java similarity index 92% rename from test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiGeneratedTest.java rename to test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiGeneratedTest.java index 13b6b0113f..fb27ee6cc7 100644 --- a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/OpenApiGeneratedTest.java +++ b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/OpenApiGeneratedTest.java @@ -1,4 +1,4 @@ -package io.micronaut.open.jaxrs; +package io.micronaut.openapi.jaxrs; import io.micronaut.core.io.ResourceLoader; import io.micronaut.test.extensions.junit5.annotation.MicronautTest; diff --git a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestController.java b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestController.java similarity index 90% rename from test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestController.java rename to test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestController.java index b50acdfdae..0f507e7ff0 100644 --- a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestController.java +++ b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestController.java @@ -1,4 +1,4 @@ -package io.micronaut.open.jaxrs; +package io.micronaut.openapi.jaxrs; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; diff --git a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestControllerTest.java b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestControllerTest.java similarity index 97% rename from test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestControllerTest.java rename to test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestControllerTest.java index 0380eb7c11..4397984f57 100644 --- a/test-suite-java-jaxrs/src/test/java/io/micronaut/open/jaxrs/TestControllerTest.java +++ b/test-suite-java-jaxrs/src/test/java/io/micronaut/openapi/jaxrs/TestControllerTest.java @@ -1,4 +1,4 @@ -package io.micronaut.open.jaxrs; +package io.micronaut.openapi.jaxrs; import io.micronaut.http.client.BlockingHttpClient; import io.micronaut.http.client.HttpClient; diff --git a/test-suite-java-spring/build.gradle b/test-suite-java-spring/build.gradle new file mode 100644 index 0000000000..f635c2575d --- /dev/null +++ b/test-suite-java-spring/build.gradle @@ -0,0 +1,52 @@ +plugins { + id("io.micronaut.build.internal.openapi-test-java") +} + +sourceSets { + test { + java { + srcDirs += "$buildDir/classes/java" + } + } +} + +dependencies { + + annotationProcessor(mnSpring.micronaut.spring.annotation) + annotationProcessor(mnSpring.micronaut.spring.web.annotation) + annotationProcessor(mnSpring.micronaut.spring.boot.annotation) + annotationProcessor(mn.micronaut.inject.java) + annotationProcessor(projects.micronautOpenapi) + + compileOnly(projects.micronautOpenapiAnnotations) + compileOnly(mn.jackson.annotations) + compileOnly(mn.micronaut.inject.java) + compileOnly(mnSerde.micronaut.serde.api) + + implementation(spring.spring.springBootStarterWeb) + implementation(spring.spring.springBootStarterValidation) + implementation(spring.spring.springBootStarterDataRest) + + testCompileOnly(projects.micronautOpenapiAnnotations) + testCompileOnly(mn.jackson.annotations) + testCompileOnly(mn.micronaut.inject.java) + testCompileOnly(mnSerde.micronaut.serde.api) + + testImplementation(spring.spring.springBootStarterTest) + testImplementation(mnTest.junit.jupiter.api) + testImplementation(projects.micronautOpenapiCommon) + + testRuntimeOnly(mnLogging.logback.classic) + testRuntimeOnly(mnTest.junit.jupiter.engine) +} + +tasks.withType(JavaCompile).configureEach { + options.encoding = "UTF-8" + options.incremental = true + options.fork = true + options.compilerArgs = [ + '-parameters', + '-Xlint:unchecked', + '-Xlint:deprecation' + ] +} diff --git a/test-suite-java-spring/openapi.properties b/test-suite-java-spring/openapi.properties new file mode 100644 index 0000000000..b0a131d727 --- /dev/null +++ b/test-suite-java-spring/openapi.properties @@ -0,0 +1 @@ +micronaut.openapi.views.spec=swagger-ui.enabled=true diff --git a/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/Application.java b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/Application.java new file mode 100644 index 0000000000..335485fc29 --- /dev/null +++ b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/Application.java @@ -0,0 +1,23 @@ +package io.micronaut.openapi.spring; + +import io.swagger.v3.oas.annotations.OpenAPIDefinition; +import io.swagger.v3.oas.annotations.info.Info; +import org.springframework.boot.Banner; +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; + +@SpringBootApplication +@OpenAPIDefinition( + info = @Info( + title = "demo", + version = "0.0" + ) +) +public class Application { + + public static void main(String[] args) { + var application = new SpringApplication(Application.class); + application.setBannerMode(Banner.Mode.OFF); + application.run(args); + } +} diff --git a/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/WebConfig.java b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/WebConfig.java new file mode 100644 index 0000000000..ee27d9ad91 --- /dev/null +++ b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/WebConfig.java @@ -0,0 +1,19 @@ +package io.micronaut.openapi.spring; + +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.EnableWebMvc; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +@Configuration +@EnableWebMvc +public class WebConfig implements WebMvcConfigurer { + + @Override + public void addResourceHandlers(ResourceHandlerRegistry registry) { + registry.addResourceHandler("/swagger-ui/**") + .addResourceLocations("classpath:/META-INF/swagger/views/swagger-ui/"); + registry.addResourceHandler("/swagger/**") + .addResourceLocations("classpath:/META-INF/swagger/"); + } +} \ No newline at end of file diff --git a/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/HelloController.java b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/HelloController.java new file mode 100644 index 0000000000..159b90dab7 --- /dev/null +++ b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/HelloController.java @@ -0,0 +1,29 @@ +package io.micronaut.openapi.spring.api; + +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +import java.util.List; +import java.util.Locale; + +@RestController +@RequestMapping("/api") +public class HelloController { + + @GetMapping + public ResponseEntity>> endpoint() { + return ResponseEntity.ok(new ResponseObject<>()); + } + + public static class ResponseObject { + + public T body; + } + + public static class Dto { + + public Locale locale; + } +} \ No newline at end of file diff --git a/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/TestController.java b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/TestController.java new file mode 100644 index 0000000000..e3e426e030 --- /dev/null +++ b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/TestController.java @@ -0,0 +1,69 @@ +package io.micronaut.openapi.spring.api; + +import io.micronaut.openapi.spring.api.dto.User; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.http.HttpStatus; +import org.springframework.http.MediaType; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PatchMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.ResponseStatus; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.SessionAttribute; + +@RestController("/user") +class TestController { + + /** + * {@summary Create post op summary.} Operation post description. + * + * @param user User request body + * + * @return created post user + */ + @PostMapping(value = "/create", + produces = {MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE}, + consumes = MediaType.APPLICATION_JSON_VALUE + ) + public User createPost(@RequestBody User user) { + user.setId(9876L); + return user; + } + + /** + * {@summary Create patch op summary.} Operation patch description. + * + * @param user User request body + */ + @PatchMapping("/create") + @ResponseStatus(code = HttpStatus.ACCEPTED) + public void createPatch(@RequestBody(required = false) User user) { + } + + @GetMapping(value = "/{userId}", produces = MediaType.TEXT_HTML_VALUE) + public String get( + @PathVariable String userId, + @RequestParam(required = false, defaultValue = "123") Integer age + ) { + return "Pong userId " + userId; + } + + @PatchMapping(value = "/patch") + public User patch( + @RequestBody User user, + @SessionAttribute(name = "mySesAttr", required = false) String sesAttr + ) { + user.setId(9876L); + return user; + } + + @GetMapping("/pageable") + public Page getSomeDTOs(Pageable pageable) { + return null; + } + +} diff --git a/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/dto/User.java b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/dto/User.java new file mode 100644 index 0000000000..6628c4b534 --- /dev/null +++ b/test-suite-java-spring/src/main/java/io/micronaut/openapi/spring/api/dto/User.java @@ -0,0 +1,32 @@ +package io.micronaut.openapi.spring.api.dto; + +public class User { + + private long id; + private String name; + private int age; + + public long getId() { + return id; + } + + public void setId(long id) { + this.id = id; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + + public int getAge() { + return age; + } + + public void setAge(int age) { + this.age = age; + } +} diff --git a/test-suite-java-spring/src/main/resources/application.yml b/test-suite-java-spring/src/main/resources/application.yml new file mode 100644 index 0000000000..6ee4dc6909 --- /dev/null +++ b/test-suite-java-spring/src/main/resources/application.yml @@ -0,0 +1,17 @@ +server: + port: 8701 + servlet: + encoding: + force-response: true + +spring: + application: + name: + main: + banner-mode: off + +logging: + pattern: + console: '%cyan(%d{HH:mm:ss.SSS}) %gray([%thread]) %highlight(%-5level) %magenta(%logger{36}) - %msg%n' + level: + root: info diff --git a/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/OpenApiExposedTest.java b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/OpenApiExposedTest.java new file mode 100644 index 0000000000..19437341fa --- /dev/null +++ b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/OpenApiExposedTest.java @@ -0,0 +1,69 @@ +package io.micronaut.openapi.spring; + +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.boot.test.context.SpringBootTest.UseMainMethod; +import org.springframework.test.context.ActiveProfiles; +import org.springframework.web.client.RestClient; + +import java.io.IOException; +import java.util.concurrent.atomic.AtomicReference; + +import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; + +@ActiveProfiles("test") +@SpringBootTest(useMainMethod = UseMainMethod.ALWAYS, classes = { + WebConfig.class, + TestConfig.class, + Application.class, +}, webEnvironment = SpringBootTest.WebEnvironment.DEFINED_PORT) +class OpenApiExposedTest { + + @Autowired + RestClient restClient; + + @Test + void testOpenApiSpecEndpoint() throws IOException { + + String openApiSpec; + try (var is = getClass().getResourceAsStream("/META-INF/swagger/demo-0.0.yml")) { + assertNotNull(is); + openApiSpec = new String(is.readAllBytes()); + } + var recievedOpenApiSpec = new AtomicReference(); + + assertDoesNotThrow(() -> { + var result = restClient.get() + .uri("/swagger/demo-0.0.yml") + .retrieve(); + + recievedOpenApiSpec.set(result.body(String.class)); + }); + + assertEquals(openApiSpec, recievedOpenApiSpec.get()); + } + + @Test + void testSwaggerUiEndpoint() throws IOException { + + String openApiSpec; + try (var is = getClass().getResourceAsStream("/META-INF/swagger/views/swagger-ui/index.html")) { + assertNotNull(is); + openApiSpec = new String(is.readAllBytes()); + } + var recievedOpenApiSpec = new AtomicReference(); + + assertDoesNotThrow(() -> { + var result = restClient.get() + .uri("/swagger-ui/index.html") + .retrieve(); + + recievedOpenApiSpec.set(result.body(String.class)); + }); + + assertEquals(openApiSpec, recievedOpenApiSpec.get()); + } +} diff --git a/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestConfig.java b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestConfig.java new file mode 100644 index 0000000000..fc0a12a23d --- /dev/null +++ b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestConfig.java @@ -0,0 +1,17 @@ +package io.micronaut.openapi.spring; + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.client.RestClient; + +@Configuration +public class TestConfig { + + @Bean + RestClient restClient(@Value("${server.port:8080}") int port) { + return RestClient.builder() + .baseUrl("http://localhost:" + port) + .build(); + } +} diff --git a/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestControllerTest.java b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestControllerTest.java new file mode 100644 index 0000000000..4eb314d3c2 --- /dev/null +++ b/test-suite-java-spring/src/test/java/io/micronaut/openapi/spring/TestControllerTest.java @@ -0,0 +1,88 @@ +package io.micronaut.openapi.spring; + +import com.fasterxml.jackson.core.JsonProcessingException; +import io.micronaut.openapi.OpenApiUtils; +import io.swagger.v3.oas.models.OpenAPI; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.ActiveProfiles; +import org.springframework.web.client.RestClient; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertTrue; + +@ActiveProfiles("test") +@SpringBootTest(useMainMethod = SpringBootTest.UseMainMethod.ALWAYS, classes = { + WebConfig.class, + TestConfig.class, + Application.class, +}, webEnvironment = SpringBootTest.WebEnvironment.DEFINED_PORT) +class TestControllerTest { + + @Autowired + RestClient restClient; + + @Test + void springOpenApiPathTest() throws JsonProcessingException { + var result = restClient.get() + .uri("/swagger/demo-0.0.yml") + .retrieve() + .body(String.class); + + var openApi = OpenApiUtils.getYamlMapper().readValue(result, OpenAPI.class); + assertNotNull(openApi.getPaths()); + + var userSchema = openApi.getComponents().getSchemas().get("User"); + assertNotNull(userSchema); + + var createPostOp = openApi.getPaths().get("/create").getPost(); + + assertEquals("Create post op summary.", createPostOp.getSummary()); + assertEquals("Create post op summary. Operation post description.", createPostOp.getDescription()); + assertNotNull(createPostOp.getRequestBody()); + + var mediaType = createPostOp.getRequestBody().getContent().get("application/json"); + assertNotNull(mediaType); + assertNotNull(mediaType.getSchema()); + assertEquals("#/components/schemas/User", mediaType.getSchema().get$ref()); + assertNotNull(createPostOp.getResponses()); + assertNotNull(createPostOp.getResponses().get("200")); + assertEquals("created post user", createPostOp.getResponses().get("200").getDescription()); + + var createPatchOp = openApi.getPaths().get("/create").getPatch(); + + assertEquals("Create patch op summary.", createPatchOp.getSummary()); + assertEquals("Create patch op summary. Operation patch description.", createPatchOp.getDescription()); + + mediaType = createPatchOp.getRequestBody().getContent().get("application/json"); + assertNotNull(mediaType); + assertNotNull(mediaType.getSchema()); + assertEquals("#/components/schemas/User", mediaType.getSchema().get$ref()); + assertNotNull(createPatchOp.getResponses()); + assertNotNull(createPatchOp.getResponses().get("202")); + assertEquals("createPatch 202 response", createPatchOp.getResponses().get("202").getDescription()); + + var userIdOp = openApi.getPaths().get("/{userId}").getGet(); + + assertNotNull(userIdOp); + + var params = userIdOp.getParameters(); + assertNotNull(params); + assertEquals(2, params.size()); + + assertEquals("userId", params.get(0).getName()); + assertEquals("path", params.get(0).getIn()); + assertTrue(params.get(0).getRequired()); + assertNotNull(params.get(0).getSchema()); + assertEquals("string", params.get(0).getSchema().getType()); + + assertEquals("age", params.get(1).getName()); + assertEquals("query", params.get(1).getIn()); + assertNotNull(params.get(1).getSchema()); + assertEquals("integer", params.get(1).getSchema().getType()); + assertEquals("int32", params.get(1).getSchema().getFormat()); + assertTrue(params.get(1).getSchema().getNullable()); + } +} diff --git a/test-suite-java-spring/src/test/resources/application-test.yml b/test-suite-java-spring/src/test/resources/application-test.yml new file mode 100644 index 0000000000..18668271c0 --- /dev/null +++ b/test-suite-java-spring/src/test/resources/application-test.yml @@ -0,0 +1,7 @@ +spring: + main: + banner-mode: off + web: + resources: + static-locations: + - classpath:META-INF/swagger/views/** diff --git a/test-suite-java-spring/src/test/resources/logback.xml b/test-suite-java-spring/src/test/resources/logback.xml new file mode 100644 index 0000000000..2ec10d0fde --- /dev/null +++ b/test-suite-java-spring/src/test/resources/logback.xml @@ -0,0 +1,13 @@ + + + + + %cyan(%d{HH:mm:ss.SSS}) %gray([%thread]) %highlight(%-5level) %magenta(%logger{36}) - %msg%n + + + + + + +