你可能尝试过写完一个接口后,自己去创建接口文档,或者修改接口后修改接口文档。多了之后,你肯定会发生一个操作,那就是忘记了修改文档或者创建文档(除非你们公司把接口文档和写接口要求得很紧密😓忘记写文档就扣工资?,否则两个分离的工作总是有可能遗漏的)。而swagger就是一个在你写接口的时候自动帮你生成接口文档的东西,只要你遵循它的规范并写一些接口的说明注解即可。
优点:
缺点
😓上面的缺点好像写的有点多,你可能会觉得swagger这个坑有点大。但其实主要是规范问题,而规范问题有时候又会提高你的代码规范性,这个就见仁见智了,你以前可能什么接口的参数都使用一个类,而现在swagger要求你分开后,某种层次上提高了你的代码规范性。
🔶注:以下代码示例基于Spring Boot。完整代码可以参考:swagger-demo
💡这里先讲添加swagger,也就是先整合进来,至于怎么使用,下面的“场景”中再讲解。
❗注意,这里的前提是已经导入了spring boot的web包。
io.springfox springfox-swagger2 2.9.2 io.springfox springfox-swagger-ui 2.9.2
要使用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.xxxx.project.tool.swagger")) //.withClassAnnotation(RestController.class) // 扫描带有指定注解的类下所有接口 //.withMethodAnnotation(PostMapping.class) // 扫描带有只当注解的方法接口 //.apis(RequestHandlerSelectors.any()) // 扫描所有 // 选择所有的API,如果你想只为部分API生成文档,可以配置这里 .paths(PathSelectors.any() .any() // 满足条件的路径,该断言总为true //.none() // 不满足条件的路径,该断言总为false(可用于生成环境屏蔽 swagger) //.ant("/user/**") // 满足字符串表达式路径 //.regex("") // 符合正则的路径 ) .build(); } /** * 用于定义API主界面的信息,比如可以声明所有的API的总标题、描述、版本 * @return */ private ApiInfo apiInfo() { Contact contact = new Contact( "我是作者姓名", // 作者姓名 "https://blog.csdn.net/", // 作者网址 "123456789@163.com"); // 作者邮箱 return new ApiInfoBuilder() .title("XX项目API") // 可以用来自定义API的主标题 .description("XX项目SwaggerAPI管理") // 可以用来描述整体的API .termsOfServiceUrl("https://www.baidu.com") // 用于定义服务的域名(跳转链接) .version("1.0") // 可以用来定义版本 .license("Swagger-的使用教程") .licenseUrl("https://blog.csdn.net") .contact(contact) .build(); // } }
如果需要配置多个组的话,就需要配置多个 docket() 方法
运行我们的Spring Boot项目,(我默认是8080端口,如果你不一样,请注意修改后续的url),访问http://localhost:8080/swagger-ui.html
然后你就可以看到一个如下的界面,由于我们暂时没有配置接口数据,所以下面显示
No operations defined in spec!
💡下面我们将介绍如何定义接口,以及在swagger UI界面中的内容。
接口有时候应该是分组的,而且大部分都是在一个controller中的,比如用户管理相关的接口应该都在UserController中,那么不同的业务的时候,应该定义/划分不同的接口组。接口组可以使用@Api来划分。
比如:
@Api(tags = "角色管理") // tags:你可以当作是这个组的名字。 @RestController public class RoleController { }
和
@Api(tags = "用户管理") // tags:你可以当作是这个组的名字。 @RestController public class UserController { }
🔵你也可以理解成基于tags来分组,就好像一些文章里面的标签一样,使用标签来分类。
🔵如果这个Controller下(接口组)下面没有接口,那么在swagger ui中是不会显示的,如果有的话就会这样显示:
使用了@Api来标注一个Controller之后,如果下面有接口,那么就会默认生成文档,但没有我们自定义的说明:
@Api(tags = "用户管理") @RestController public class UserController { // 注意,对于swagger,不要使用@RequestMapping, // 因为@RequestMapping支持任意请求方式,swagger会为这个接口生成7种请求方式的接口文档 @GetMapping("/info") public String info(String id){ return "aaa"; } }
我们可以使用@ApiOperation来描述接口,比如:
@ApiOperation(value = "用户测试",notes = "用户测试notes") @GetMapping("/test") public String test(String id){ return "test"; }
常用配置项:
上面使用了@ApiOperation来了描述接口,但其实还缺少接口请求参数的说明,下面我们分场景来讲。
🔵注意一下,对于GET方式,swagger不推荐使用body方式来传递数据,也就是不希望在GET方式时使用json、form-data等方式来传递,这时候最好使用路径参数或者url参数。(😓虽然POSTMAN等是支持的),所以如果接口传递的数据是json或者form-data方式的,还是使用POST方式好。
此时我们需要使用@ApiModel来标注实体类,然后在接口中定义入参为实体类即可:
// 先使用@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; // 此处省略入参赋值时需要的getter,setter,swagger也需要这个 }
定义成入参:
@ApiOperation(value = "登录接口",notes = "登录接口的说明") @PostMapping("/login") public LoginForm login(@RequestBody LoginForm loginForm){ return loginForm; }
效果:
(再说一次:对于GET方式,swagger不推荐使用body方式来传递数据,所以虽然Spring MVC可以自动封装参数,但对于GET请求还是不要使用form-data,json等方式传递参数,除非你使用Postman来测试接口,swagger在线测试是不支持这个操作的)
对于非实体类参数,可以使用@ApiImplicitParams和@ApiImplicitParam来声明请求参数。
@ApiImplicitParams用在方法头上,@ApiImplicitParam定义在@ApiImplicitParams里面,一个@ApiImplicitParam对应一个参数。
@ApiImplicitParam常用配置项:
【如果你的接口是form-data,x-www-form-urlencoded的时候可能不能使用swagger页面API调试,但可以在后面讲到基于BootstrapUI的swagger增强中调试,基于BootstrapUI的swagger支持指定form-data或x-www-form-urlencoded】
// 使用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; }
// 使用路径参数 @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传递参数 @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(); return originalFilename; } // 多个文件上传时,**swagger只能测试单文件上传** @ApiOperation(value = "上传多个文件",notes = "上传多个文件") @PostMapping(value = "/upload2",consumes = "multipart/*", headers = "content-type=multipart/form-data") public String upload2(@ApiParam(value = "图片文件", required = true,allowMultiple = true)MultipartFile[] uploadFile){ StringBuffer sb = new StringBuffer(); for (int i = 0; i < uploadFile.length; i++) { System.out.println(uploadFile[i].getOriginalFilename()); sb.append(uploadFile[i].getOriginalFilename()); sb.append(","); } return sb.toString(); } // 既有文件,又有参数 @ApiOperation(value = "既有文件,又有参数",notes = "既有文件,又有参数") @PostMapping(value = "/upload3") @ApiImplicitParams({ @ApiImplicitParam(name = "name", value = "图片新名字", required = true ) }) public String upload3(@ApiParam(value = "图片文件", required = true)MultipartFile uploadFile, String name){ String originalFilename = uploadFile.getOriginalFilename(); return originalFilename+":"+name; } ~~~ ## 定义接口响应 ### 响应是实体类: 前面在定义接口请求参数的时候有提到使用`@ApiModel`来标注类,如果接口返回了这个类,那么这个类上的说明也会作为响应的说明: ```java // 返回被@ApiModel标注的类对象 @ApiOperation(value = "实体类响应",notes = "返回数据为实体类的接口") @PostMapping("/role1") public LoginForm role1(@RequestBody LoginForm loginForm){ return loginForm; }
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" + " }"; }
你可能会觉得现在这个UI不是很好看,现在有一些第三方提供了一些Swagger UI增强,比较流行的是swagger-bootstrap-ui,我们这里以swagger-bootstrap-ui为例。
1.添加依赖包:
io.springfox springfox-swagger2 2.9.2 io.springfox springfox-swagger-ui 2.9.2 com.github.xiaoymin swagger-bootstrap-ui 1.8.7
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-data或x-www-form-urlencoded:
3.支持复制单个API文档和导出全部API文档:
在Spring Boot整合Spring Security和Swagger的时候,需要配置拦截的路径和放行的路径,注意是放行以下几个路径。
.antMatchers("/swagger**/**").permitAll() .antMatchers("/webjars/**").permitAll() .antMatchers("/v2/**").permitAll() .antMatchers("/doc.html").permitAll() // 如果你用了bootstarp的Swagger UI界面,加一个这个。
在swagger中只支持了简单的调试,但对于一些接口,我们测试的时候可能需要把token信息写到header中,目前好像没看到可以自定义加请求头的地方?
💡方法一:如果你使用了Swagger BootstrapUI,那么你可以在“文档管理”中增加全局参数,这包括了添加header参数。
💡方法二:在swagger配置类中增加全局参数配置:
//如果有额外的全局参数,比如说请求头参数,可以这样添加 ParameterBuilder parameterBuilder = new ParameterBuilder(); Listparameters = new ArrayList (); 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; }
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
想到swagger3已经更新了,于是想着尝鲜使用下,的确省去了很多配置。
这里写下使用流程。
参考:Springfox 3.0.0(包含springfox-swagger2-3.0.0)即OpenAPI 3的发布与系统集成
官网地址:https://swagger.io/resources/open-api/
Swagger3是在Swagger2上做了大版本升级,使用方式和Swagger2大体一致,只是需要注意区别配置地方,Swagger3使用相关依赖版本如下:
区别 | swagger2 | swagger3 |
---|---|---|
依赖包 | springfox-swagger2、springfox-swagger-ui | springfox-boot-starter |
启用方式 | @EnableSwagger2 | @EnableOpenApi |
访问方式 | ip:port/swagger-ui.html | ip:port/swagger-ui/index.html |
Doucument类型 | DocumentationType.SWAGGER_2 | DocumentationType.OAS_30 |
优化:openapi3依赖添加,现在非常方便,不像以往2.9.2版本需要好几个依赖,现在只需添加一个
io.springfox springfox-boot-starter 3.0.0
推荐配置文件和代码分离
注解的更改:由之前的@Swagger2更改为@EnableOpenApi
Docket构造函数中的DocumentationType指向更改:由之前的DocumentationType.SWAGGER_2 更改为DocumentationType.OAS_30
对于扫描路径部分可以自行选择(当项目人多的时候可以自由扩展,人少就自行安排)
paths(PathSelectors.regex(“asds”)
.or(PathSelectors.regex(“adsd”))
.or(…)
)
此外还具备swagger进行授权配置,在开发环境的基础上进一步封装
@PropertySource(value = {"classpath:/config/Swagger3.properties"}) @EnableOpenApi @Configuration public class Swagger3Config { @Value("${nature.restApi.enabled}") private Boolean enable; @Value("${application.name}") private String applicationName; @Value("${application.version}") private String applicationVersion; @Value("${application.description}") private String applicationDescription; @Value("${application.group01}") private String applicationGroup01; @Value("${application.apis.selector}") private String selector; @Bean public Docket docket() { return new Docket(DocumentationType.OAS_30) .apiInfo(apiInfo()) .groupName(applicationGroup01) .select() .apis(RequestHandlerSelectors.basePackage(selector)) .paths(PathSelectors.any()) .build(); } private ApiInfo apiInfo() { return new ApiInfoBuilder() .title(applicationName) .description(applicationDescription) .version(applicationVersion) .build(); } } /** 对应的配置文件信息 * 是否开启swagger,开发环境打开,线上关闭 * nature.restApi.enabled=true * 项目名称 * application.name=SecKill-应用服务API * 项目版本信息 * application.version=1.0 * 项目描述信息 * application.description=秒杀-API接口文档 * 分组信息(可自行添加) * application.group01=lyh * 扫描路径选择 application.apis.selector=com.gan.springseckill.controller */