Spring Boot整合swagger使用教程


Swagger 的介绍

🔶你可能尝试过写完一个接口后,自己去创建接口文档,或者修改接口后修改接口文档。多了之后,你肯定会发生一个操作,那就是忘记了修改文档或者创建文档(除非你们公司把接口文档和写接口要求得很紧密😓忘记写文档就扣工资?,否则两个分离的工作总是有可能遗漏的)。而 swagger 就是一个在你写接口的时候自动帮你生成接口文档的东西,只要你遵循它的规范并写一些接口的说明注解即可。


优点与缺点

🔶优点:

  • 自动生成文档,只需要在接口中使用注解进行标注,就能生成对应的接口文档。
  • 自动更新文档,由于是动态生成的,所以如果你修改了接口,文档也会自动对应修改(如果你也更新了注解的话)。这样就不会发送我修改了接口,却忘记更新接口文档的情况。
  • 支持在线调试,swagger 提供了在线调用接口的功能。

🔶缺点:

  • 不能创建测试用例,所以他暂时不能帮你处理完所有的事情。他只能提供一个简单的在线调试,如果你想存储你的测试用例,可以使用 Postman 或者 YAPI 这样支持创建测试用户的功能。
  • 要遵循一些规范,它不是任意规范的。比如说,你可能会返回一个 json 数据,而这个数据可能是一个 Map 格式的,那么我们此时不能标注这个 Map 格式的返回数据的每个字段的说明,而如果它是一个实体类的话,我们可以通过标注类的属性来给返回字段加说明。也比如说,对于 swagger,不推荐在使用 GET 方式提交数据的时候还使用 Body,仅推荐使用 query 参数、header 参数或者路径参数,当然了这个限制只适用于在线调试。
  • 没有接口文档更新管理,虽然一个接口更新之后,可能不会关心旧版的接口信息,但你“可能”想看看旧版的接口信息,例如有些灰度更新发布的时候可能还会关心旧版的接口。那么此时只能由后端去看看有没有注释留下了,所以可以考虑接口文档大更新的时候注释旧版的,然后写下新版的。【当然这个问题可以通过导出接口文档来对比。】
  • 虽然现在 Java 的实体类中有不少模型,po,dto,vo 等,模型的区分是为了屏蔽一些多余参数,比如一个用户登录的时候只需要 username,password,但查权限的时候需要连接上权限表的信息,而如果上述两个操作都是使用了 User 这个实体的话,在文档中就会自动生成了多余的信息,这就要求了你基于模型来创建多个实体类,比如登录的时候一个 LoginForm,需要用户 - 权限等信息的时候才使用 User 类。(当然了,这个问题等你会 swagger 之后你就大概就会怎么规避这个问题了。)

😓上面的缺点好像写的有点多,你可能会觉得 swagger 这个坑有点大。但其实主要是规范问题,而规范问题有时候又会提高你的代码规范性,这个就见仁见智了,你以前可能什么接口的参数都使用一个类,而现在 swagger 要求你分开后,某种层次上提高了你的代码规范性。


🔶注:以下代码示例基于 Spring Boot。完整代码可以参考:swagger-demo



添加 swagger

💡这里先讲添加 swagger,也就是先整合进来,至于怎么使用,下面的“场景”中再讲解。


1. 添加依赖包:

❗注意,这里的前提是已经导入了 spring boot 的 web 包。

复制代码
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger2</artifactId>
            <version>2.9.2</version>
        </dependency>
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger-ui</artifactId>
            <version>2.9.2</version>
        </dependency>

2. 配置 Swagger:

要使用 swagger,我们必须对 swagger 进行配置,我们需要创建一个 swagger 的配置类,比如可以命名为 SwaggerConfig.java

复制代码
package com.example.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import springfox.documentation.builders.ApiInfoBuilder;
import springfox.documentation.builders.PathSelectors;
import springfox.documentation.builders.RequestHandlerSelectors;
import springfox.documentation.service.ApiInfo;
import springfox.documentation.spi.DocumentationType;
import springfox.documentation.spring.web.plugins.Docket;
import springfox.documentation.swagger2.annotations.EnableSwagger2;

@Configuration // 标明是配置类
@EnableSwagger2 // 开启 swagger 功能
public class SwaggerConfig {
@Bean
public Docket createRestApi() {
return new Docket(DocumentationType.SWAGGER_2) // DocumentationType.SWAGGER_2 固定的,代表 swagger2
// .groupName("分布式任务系统") // 如果配置多个文档的时候,那么需要配置 groupName 来分组标识
.apiInfo(apiInfo()) // 用于生成 API 信息
.select() // select() 函数返回一个 ApiSelectorBuilder 实例, 用来控制接口被 swagger 做成文档
.apis(RequestHandlerSelectors.basePackage("com.example.controller")) // 用于指定扫描哪个包下的接口
.paths(PathSelectors.any())// 选择所有的 API, 如果你想只为部分 API 生成文档,可以配置这里
.build();
}

<span class="hljs-comment">/**
 * 用于定义API主界面的信息,比如可以声明所有的API的总标题、描述、版本
 * <span class="hljs-doctag">@return</span>
 */</span>
<span class="hljs-keyword">private</span> ApiInfo <span class="hljs-title function_">apiInfo</span><span class="hljs-params">()</span> {
    <span class="hljs-keyword">return</span> <span class="hljs-keyword">new</span> <span class="hljs-title class_">ApiInfoBuilder</span>()
            .title(<span class="hljs-string">"XX项目API"</span>) <span class="hljs-comment">//  可以用来自定义API的主标题</span>
            .description(<span class="hljs-string">"XX项目SwaggerAPI管理"</span>) <span class="hljs-comment">// 可以用来描述整体的API</span>
            .termsOfServiceUrl(<span class="hljs-string">""</span>) <span class="hljs-comment">// 用于定义服务的域名</span>
            .version(<span class="hljs-string">"1.0"</span>) <span class="hljs-comment">// 可以用来定义版本。</span>
            .build(); <span class="hljs-comment">//</span>
}

}


3. 测试

运行我们的 Spring Boot 项目,(我默认是 8080 端口,如果你不一样,请注意修改后续的 url),访问http://localhost:8080/swagger-ui.html
然后你就可以看到一个如下的界面,由于我们暂时没有配置接口数据,所以下面显示No operations defined in spec!

20200711013419


💡下面我们将介绍如何定义接口,以及在 swagger UI 界面中的内容。



场景:

定义接口组

接口有时候应该是分组的,而且大部分都是在一个 controller 中的,比如用户管理相关的接口应该都在 UserController 中,那么不同的业务的时候,应该定义 / 划分不同的接口组。接口组可以使用@Api来划分。
比如:

复制代码
@Api(tags = "角色管理") //  tags:你可以当作是这个组的名字。
@RestController
public class RoleController {
}

复制代码
@Api(tags = "用户管理") //  tags:你可以当作是这个组的名字。
@RestController
public class UserController {
}

🔵你也可以理解成基于 tags 来分组,就好像一些文章里面的标签一样,使用标签来分类。
🔵如果这个 Controller 下(接口组)下面没有接口,那么在 swagger ui 中是不会显示的,如果有的话就会这样显示:
20200712022545


定义接口

使用了@Api来标注一个 Controller 之后,如果下面有接口,那么就会默认生成文档,但没有我们自定义的说明:

复制代码
@Api(tags = "用户管理")
@RestController
public class UserController {
    // 注意,对于 swagger,不要使用 @RequestMapping,
    // 因为 @RequestMapping 支持任意请求方式,swagger 会为这个接口生成 7 种请求方式的接口文档
    @GetMapping("/info") 
    public String info(String id){
        return "aaa";
    }
}

20200711015840

20200711020149


我们可以使用@ApiOperation来描述接口,比如:

复制代码
    @ApiOperation(value = "用户测试",notes = "用户测试 notes")
    @GetMapping("/test")
    public String test(String id){
        return "test";
    }

20200711021112
常用配置项:

  • value:可以当作是接口的简称
  • notes:接口的描述
  • tags:可以额外定义接口组,比如这个接口外层已经有@Api(tags = "用户管理"),将接口划分到了“用户管理”中,但你可以额外的使用 tags,例如tags = "角色管理"让角色管理中也有这个接口文档。

定义接口请求参数

上面使用了@ApiOperation来了描述接口,但其实还缺少接口请求参数的说明,下面我们分场景来讲。
🔵注意一下,对于 GET 方式,swagger 不推荐使用 body 方式来传递数据,也就是不希望在 GET 方式时使用 json、form-data 等方式来传递,这时候最好使用路径参数或者 url 参数。(😓虽然 POSTMAN 等是支持的),所以如果接口传递的数据是 json 或者 form-data 方式的,还是使用 POST 方式好。

场景一:请求参数是实体类。

此时我们需要使用@ApiModel来标注实体类,然后在接口中定义入参为实体类即可:

  • @ApiModel:用来标类
    • 常用配置项:
      • value:实体类简称
      • description:实体类说明
  • @ApiModelProperty:用来描述类的字段的意义。
    • 常用配置项:
      • value:字段说明
      • example:设置请求示例(Example Value)的默认值,如果不配置,当字段为 string 的时候,此时请求示例中默认值为 "".
      • name:用新的字段名来替代旧的字段名。
      • allowableValues:限制值得范围,例如{1,2,3}代表只能取这三个值;[1,5]代表取 1 到 5 的值;(1,5)代表 1 到 5 的值,不包括 1 和 5;还可以使用 infinity 或 -infinity 来无限值,比如[1, infinity]代表最小值为 1,最大值无穷大。
      • required:标记字段是否必填,默认是 false,
      • hidden:用来隐藏字段,默认是 false,如果要隐藏需要使用 true,因为字段默认都会显示,就算没有@ApiModelProperty
复制代码
// 先使用 @ApiModel 来标注类
@ApiModel(value="用户登录表单对象",description="用户登录表单对象")
public class LoginForm {
    // 使用 ApiModelProperty 来标注字段属性。
    @ApiModelProperty(value = "用户名",required = true,example = "root")
    private String username;
    @ApiModelProperty(value = "密码",required = true,example = "123456")
    private String password;
<span class="hljs-comment">// 此处省略入参赋值时需要的getter,setter,swagger也需要这个</span>

}

定义成入参:

复制代码
    @ApiOperation(value = "登录接口",notes = "登录接口的说明")
    @PostMapping("/login")
    public LoginForm login(@RequestBody LoginForm loginForm){
        return loginForm;
    }

效果:

20200711181038


场景二:请求参数是非实体类。

再说一次:对于 GET 方式,swagger 不推荐使用 body 方式来传递数据,所以虽然 Spring MVC 可以自动封装参数,但对于 GET 请求还是不要使用 form-data,json 等方式传递参数,除非你使用 Postman 来测试接口,swagger 在线测试是不支持这个操作的
对于非实体类参数,可以使用@ApiImplicitParams@ApiImplicitParam来声明请求参数。
@ApiImplicitParams用在方法头上,@ApiImplicitParam定义在@ApiImplicitParams里面,一个@ApiImplicitParam对应一个参数。
@ApiImplicitParam常用配置项:

  • name:用来定义参数的名字,也就是字段的名字, 可以与接口的入参名对应。如果不对应,也会生成,所以可以用来定义额外参数!
  • value:用来描述参数
  • required:用来标注参数是否必填
  • paramType 有 path,query,body,form,header 等方式,但对于对于非实体类参数的时候,常用的只有 path,query,header;body 和 form 是不常用的。body 不适用于多个零散参数的情况,只适用于 json 对象等情况。【如果你的接口是form-data,x-www-form-urlencoded的时候可能不能使用 swagger 页面 API 调试,但可以在后面讲到基于 BootstrapUI 的 swagger 增强中调试,基于 BootstrapUI 的 swagger 支持指定form-datax-www-form-urlencoded

示例一:声明入参是 URL 参数

复制代码
    // 使用 URL query 参数
    @ApiOperation(value = "登录接口 2",notes = "登录接口的说明 2")
    @ApiImplicitParams({
            @ApiImplicitParam(name = "username",// 参数名字
                    value = "用户名",// 参数的描述
                    required = true,// 是否必须传入
                    //paramType 定义参数传递类型:有 path,query,body,form,header
                    paramType = "query"
                    )
            ,
            @ApiImplicitParam(name = "password",// 参数名字
                    value = "密码",// 参数的描述
                    required = true,// 是否必须传入
                    paramType = "query"
                    )})
    @PostMapping(value = "/login2")
    public LoginForm login2(String username,String password){
        System.out.println(username+":"+password);
        LoginForm loginForm = new LoginForm();
        loginForm.setUsername(username);
        loginForm.setPassword(password);
        return loginForm;
    }

示例二:声明入参是 URL 路径参数

复制代码
    // 使用路径参数
    @PostMapping("/login3/{id1}/{id2}")
    @ApiOperation(value = "登录接口 3",notes = "登录接口的说明 3")
    @ApiImplicitParams({
            @ApiImplicitParam(name = "id1",// 参数名字
                    value = "用户名",// 参数的描述
                    required = true,// 是否必须传入
                    //paramType 定义参数传递类型:有 path,query,body,form,header
                    paramType = "path"
            )
            ,
            @ApiImplicitParam(name = "id2",// 参数名字
                    value = "密码",// 参数的描述
                    required = true,// 是否必须传入
                    paramType = "path"
            )})
    public String login3(@PathVariable Integer id1,@PathVariable Integer id2){
        return id1+":"+id2;
    }

示例三:声明入参是 header 参数

复制代码
    // 用 header 传递参数
    @PostMapping("/login4")
    @ApiOperation(value = "登录接口 4",notes = "登录接口的说明 4")
    @ApiImplicitParams({
            @ApiImplicitParam(name = "username",// 参数名字
                    value = "用户名",// 参数的描述
                    required = true,// 是否必须传入
                    //paramType 定义参数传递类型:有 path,query,body,form,header
                    paramType = "header"
            )
            ,
            @ApiImplicitParam(name = "password",// 参数名字
                    value = "密码",// 参数的描述
                    required = true,// 是否必须传入
                    paramType = "header"
            )})
    public String login4( @RequestHeader String username,
                          @RequestHeader String password){
        return username+":"+password;
    }

示例四:声明文件上传参数

复制代码
    // 有文件上传时要用 @ApiParam,用法基本与 @ApiImplicitParam 一样,不过 @ApiParam 用在参数上
    // 或者你也可以不注解,swagger 会自动生成说明
    @ApiOperation(value = "上传文件",notes = "上传文件")
    @PostMapping(value = "/upload")
    public String upload(@ApiParam(value = "图片文件", required = true)MultipartFile uploadFile){
        String originalFilename = uploadFile.getOriginalFilename();
    <span class="hljs-keyword">return</span> originalFilename;
}


<span class="hljs-comment">// 多个文件上传时,**swagger只能测试单文件上传**</span>
<span class="hljs-meta">@ApiOperation(value = "上传多个文件",notes = "上传多个文件")</span>
<span class="hljs-meta">@PostMapping(value = "/upload2",consumes = "multipart/*", headers = "content-type=multipart/form-data")</span>
<span class="hljs-keyword">public</span> String <span class="hljs-title function_">upload2</span><span class="hljs-params">(<span class="hljs-meta">@ApiParam(value = "图片文件", required = true,allowMultiple = true)</span>MultipartFile[] uploadFile)</span>{
    <span class="hljs-type">StringBuffer</span> <span class="hljs-variable">sb</span> <span class="hljs-operator">=</span> <span class="hljs-keyword">new</span> <span class="hljs-title class_">StringBuffer</span>();
    <span class="hljs-keyword">for</span> (<span class="hljs-type">int</span> <span class="hljs-variable">i</span> <span class="hljs-operator">=</span> <span class="hljs-number">0</span>; i &lt; uploadFile.length; i++) {
        System.out.println(uploadFile[i].getOriginalFilename());
        sb.append(uploadFile[i].getOriginalFilename());
        sb.append(<span class="hljs-string">","</span>);
    }
    <span class="hljs-keyword">return</span> sb.toString();
}



<span class="hljs-comment">// 既有文件,又有参数</span>
<span class="hljs-meta">@ApiOperation(value = "既有文件,又有参数",notes = "既有文件,又有参数")</span>
<span class="hljs-meta">@PostMapping(value = "/upload3")</span>
<span class="hljs-meta">@ApiImplicitParams({
        @ApiImplicitParam(name = "name",
                value = "图片新名字",
                required = true
        )
})</span>
<span class="hljs-keyword">public</span> String <span class="hljs-title function_">upload3</span><span class="hljs-params">(<span class="hljs-meta">@ApiParam(value = "图片文件", required = true)</span>MultipartFile uploadFile,
                      String name)</span>{
    <span class="hljs-type">String</span> <span class="hljs-variable">originalFilename</span> <span class="hljs-operator">=</span> uploadFile.getOriginalFilename();

    <span class="hljs-keyword">return</span> originalFilename+<span class="hljs-string">":"</span>+name;
}


定义接口响应

定义接口响应,是方便查看接口文档的人能够知道接口返回的数据的意义。

响应是实体类:

前面在定义接口请求参数的时候有提到使用@ApiModel来标注类,如果接口返回了这个类,那么这个类上的说明也会作为响应的说明:

复制代码

    // 返回被 @ApiModel 标注的类对象
    @ApiOperation(value = "实体类响应",notes = "返回数据为实体类的接口")
    @PostMapping("/role1")
    public LoginForm role1(@RequestBody LoginForm loginForm){
        return loginForm;
    }

20200712000406


响应是非实体类:

swagger 无法对非实体类的响应进行详细说明,只能标注响应码等信息。是通过@ApiResponses@ApiResponse来实现的。
@ApiResponses@ApiResponse可以与@ApiModel一起使用。

复制代码
    // 其他类型的, 此时不能增加字段注释,所以其实 swagger 推荐使用实体类
    @ApiOperation(value = "非实体类",notes = "非实体类")
    @ApiResponses({@ApiResponse(code=200,message = "调用成功"),
            @ApiResponse(code=401,message = "无权限")
    }
    )
    @PostMapping("/role2")
    public String role2(){
        return "{\n" +
                "name:\" 广东 \",\n" +
                "citys:{\n" +
                "city:[\" 广州 \",\" 深圳 \",\" 珠海 \"]\n" +
                "}\n" +
                "}";
    }

20200712013503



Swagger UI 增强

你可能会觉得现在这个 UI 不是很好看,现在有一些第三方提供了一些 Swagger UI 增强,比较流行的是swagger-bootstrap-ui,我们这里以swagger-bootstrap-ui为例。

UI 对比:

20200712013653

20200712013723


使用

1. 添加依赖包:

复制代码
        <!-- 引入 swagger-->
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger2</artifactId>
            <version>2.9.2</version>
        </dependency>
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger-ui</artifactId>
            <version>2.9.2</version>
        </dependency>
        <!-- 引入 swagger-bootstrap-ui 依赖包 -->
        <dependency>
            <groupId>com.github.xiaoymin</groupId>
            <artifactId>swagger-bootstrap-ui</artifactId>
            <version>1.8.7</version>
        </dependency>

2. 在 swagger 配置类中增加注解@EnableSwaggerBootstrapUI:

复制代码
@Configuration // 标明是配置类
@EnableSwagger2 // 开启 swagger 功能
@EnableSwaggerBootstrapUI // 开启 SwaggerBootstrapUI
public class SwaggerConfig {
    // 省略配置内容
}

3. 访问 API:http://localhost:8080/doc.html,即可预览到基于 bootstarp 的 Swagger UI 界面。


优点

1.🤔界面好看了一点

2. 上面说过了,基于 BootstrapUI 的 swagger 支持指定form-datax-www-form-urlencoded
20200712024858

3. 支持复制单个 API 文档和导出全部 API 文档:
20200712025020

20200712025044



整合 Spring Security 注意

在 Spring Boot 整合 Spring Security 和 Swagger 的时候,需要配置拦截的路径和放行的路径,注意是放行以下几个路径。

复制代码
.antMatchers("/swagger**/**").permitAll()
.antMatchers("/webjars/**").permitAll()
.antMatchers("/v2/**").permitAll()
.antMatchers("/doc.html").permitAll() // 如果你用了 bootstarp 的 Swagger UI 界面,加一个这个。


对于 token 的处理

在 swagger 中只支持了简单的调试,但对于一些接口,我们测试的时候可能需要把 token 信息写到 header 中,目前好像没看到可以自定义加请求头的地方?
💡方法一:
  如果你使用了 Swagger BootstrapUI,那么你可以在“文档管理”中增加全局参数,这包括了添加 header 参数。

💡方法二:在 swagger 配置类中增加全局参数配置:

复制代码
        // 如果有额外的全局参数,比如说请求头参数,可以这样添加
        ParameterBuilder parameterBuilder = new ParameterBuilder();
        List<Parameter> parameters = new ArrayList<Parameter>();
        parameterBuilder.name("authorization").description("令牌")
                .modelRef(new ModelRef("string")).parameterType("header").required(false).build();
        parameters.add(parameterBuilder.build());
        return new Docket(DocumentationType.SWAGGER_2)  // DocumentationType.SWAGGER_2 固定的,代表 swagger2
                .apiInfo(apiInfo()) // 用于生成 API 信息
                .select() // select() 函数返回一个 ApiSelectorBuilder 实例, 用来控制接口被 swagger 做成文档
                .apis(RequestHandlerSelectors.basePackage("com.example.controller")) // 用于指定扫描哪个包下的接口
                .paths(PathSelectors.any())// 选择所有的 API, 如果你想只为部分 API 生成文档,可以配置这里
                .build().globalOperationParameters(parameters);

💡方法三:使用@ApiImplicitParams来额外标注一个请求头参数,例如:

复制代码
    // 如果需要额外的参数,非本方法用到,但过滤器要用, 类似于权限 token
    @PostMapping("/login6")
    @ApiOperation(value = "带 token 的接口",notes = "带 token 的接口")
    @ApiImplicitParams({
            @ApiImplicitParam(name = "authorization",// 参数名字
                    value = "授权 token",// 参数的描述
                    required = true,// 是否必须传入
                    paramType = "header"
            )
            ,
            @ApiImplicitParam(name = "username",// 参数名字
                    value = "用户名",// 参数的描述
                    required = true,// 是否必须传入
                    paramType = "query"
            )})
    public String login6(String username){
        return username;
    }


Swagger 的安全管理

1. 如果你整合了权限管理,可以给 swagger 加上权限管理,要求访问 swagger 页面输入用户名和密码,这些是 spring security 和 shiro 的事了,这里不讲。


2. 如果你仅仅是不想在正式环境中可以访问,可以在正式环境中关闭 Swagger 自动配置,这就不会有 swagger 页面了。使用@Profile({"dev","test"})注解来限制只在 dev 或者 test 下启用 Swagger 自动配置。
然后在 Spring Boot 配置文件中修改当前 profilespring.profiles.active=release,重启之后,此时无法访问http://localhost:8080/swagger-ui.html



作者:progor
本文为作者原创,转载请注明出处