Usage of Swagger 2.0 in Spring Boot Applications to document APIs
As IBM VP Angel Diaz stated in an interview on SearchCloudComputing “Swagger is the way most developers describe [REST] APIs”. With version 2.0 many important features like extensibility have been added, there is a big community and many developers are using it by now. Additionally there is work going on to create an open governance model around the Swagger specification under the Linux Foundation as part of the Open API Initiative.
What I like most about Swagger is the ability to document APIs directly in the (Java) source code via annotations so that documentation and actual API implementations are not out of synch.
IBM uses Swagger for the API documentation of several products and services internally. Furthermore the API management service in IBM Bluemix can import Swagger 2.0 definitions to manage your APIs easily.
The Java Spring framework has a lot of traction in the enterprise community these days. In order to use Swagger in Spring Boot applications I tried Springfox which is not part of Swagger core but it integrates nicely in Spring and supports the core Swagger annotations.
Below I extended the Spring sample Building a RESTful Web Service with Swagger annotations.
First you need to define the dependencies to the Springfox and Swagger libraries, in my case in Maven.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
| <? xml version = "1.0" encoding = "UTF-8" ?> < project xmlns = "http://maven.apache.org/POM/4.0.0" xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" < modelVersion >4.0.0</ modelVersion > < groupId >org.springframework</ groupId > < artifactId >gs-rest-service</ artifactId > < version >0.1.0</ version > < parent > < groupId >org.springframework.boot</ groupId > < artifactId >spring-boot-starter-parent</ artifactId > < version >1.3.0.RELEASE</ version > </ parent > < dependencies > < dependency > < groupId >org.springframework.boot</ groupId > < artifactId >spring-boot-starter-web</ artifactId > </ dependency > < dependency > < groupId >io.springfox</ groupId > < artifactId >springfox-swagger-ui</ artifactId > < version >2.2.2</ version > < scope >compile</ scope > </ dependency > < dependency > < groupId >io.springfox</ groupId > < artifactId >springfox-swagger2</ artifactId > < version >2.2.2</ version > < scope >compile</ scope > </ dependency > </ dependencies > < properties > < java.version >1.8</ java.version > </ properties > < build > < plugins > < plugin > < groupId >org.springframework.boot</ groupId > < artifactId >spring-boot-maven-plugin</ artifactId > </ plugin > </ plugins > </ build > < repositories > < repository > < id >spring-releases</ id > </ repository > </ repositories > < pluginRepositories > < pluginRepository > < id >spring-releases</ id > </ pluginRepository > </ pluginRepositories > </ project > |
Then I enabled Swagger in the boot application.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
| package hello; import static springfox.documentation.builders.PathSelectors.regex; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import com.google.common.base.Predicate; import springfox.documentation.builders.ApiInfoBuilder; import springfox.documentation.service.ApiInfo; import springfox.documentation.spi.DocumentationType; import springfox.documentation.spring.web.plugins.Docket; import springfox.documentation.swagger2.annotations.EnableSwagger2; @SpringBootApplication @EnableSwagger2 @ComponentScan ( "hello" ) public class Application { public static void main(String[] args) { SpringApplication.run(Application. class , args); } @Bean public Docket newsApi() { return new Docket(DocumentationType.SWAGGER_2) .groupName( "greetings" ) .apiInfo(apiInfo()) .select() .paths(regex( "/greeting.*" )) .build(); } private ApiInfo apiInfo() { return new ApiInfoBuilder() .title( "Spring REST Sample with Swagger" ) .description( "Spring REST Sample with Swagger" ) .contact( "Niklas Heidloff" ) .license( "Apache License Version 2.0" ) .version( "2.0" ) .build(); } } |
In the controller I used Swagger annotations to document the API.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
| package hello; import java.util.concurrent.atomic.AtomicLong; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; import io.swagger.annotations.ApiImplicitParam; import io.swagger.annotations.ApiImplicitParams; import io.swagger.annotations.ApiOperation; import io.swagger.annotations.ApiResponse; import io.swagger.annotations.ApiResponses; import io.swagger.annotations.ResponseHeader; @RestController public class GreetingController { private static final String template = "Hello, %s!" ; private final AtomicLong counter = new AtomicLong(); @ApiOperation (value = "getGreeting" , nickname = "getGreeting" ) @RequestMapping (method = RequestMethod.GET, path= "/greeting" , produces = "application/json" ) @ApiImplicitParams ({ @ApiImplicitParam (name = "name" , value = "User's name" , required = false , dataType = "string" , paramType = "query" , defaultValue= "Niklas" ) }) @ApiResponses (value = { @ApiResponse (code = 200 , message = "Success" , response = Greeting. class ), @ApiResponse (code = 401 , message = "Unauthorized" ), @ApiResponse (code = 403 , message = "Forbidden" ), @ApiResponse (code = 404 , message = "Not Found" ), @ApiResponse (code = 500 , message = "Failure" )}) public Greeting greeting( @RequestParam (value= "name" , defaultValue= "World" ) String name) { return new Greeting(counter.incrementAndGet(), String.format(template, name)); } } |
In the sample resource I documented parameters.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
| package hello; import com.fasterxml.jackson.annotation.JsonProperty; import io.swagger.annotations.ApiModelProperty; public class Greeting { private final long id; private final String content; public Greeting( long id, String content) { this .id = id; this .content = content; } public long getId() { return id; } @JsonProperty (required = true ) @ApiModelProperty (notes = "The name of the user" , required = true ) public String getContent() { return content; } } |
Here is the outcome in the API explorer.
댓글 없음:
댓글 쓰기