前言

Swagger是用来描述和文档化RESTful API的一个项目。Swagger Spec是一套规范,定义了该如何去描述一个RESTful API。类似的项目还有RAML、API Blueprint。 根据Swagger Spec来描述RESTful API的文件称之为Swagger specification file,它使用JSON来表述,也支持作为JSON支持的YAML。

Swagger specification file可以用来给swagger-ui生成一个Web的可交互的文档页面,以可以用swagger2markup生成静态文档,也可用使用swagger-codegen生成客户端代码。总之有了有个描述API的JSON文档之后,可以做各种扩展。

Swagger specification file可以手动编写,swagger-editor为了手动编写的工具提供了预览的功能。但是实际写起来也是非常麻烦的,同时还得保持代码和文档的两边同步。于是针对各种语言的各种框架都有一些开源的实现来辅助自动生成这个`Swagger specification file。

swagger-core是一个Java的实现,现在支持JAX-RS。swagger-annotation定义了一套注解给用户用来描述API。
spring-fox也是一个Java的实现,它支持Spring MVC, 它也支持swagger-annotation定义的部分注解。

使用

添加依赖

在pom文件添加:

<swagger.version>2.7.0</swagger.version><dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>${swagger.version}</version></dependency><dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger-ui</artifactId><version>${swagger.version}</version></dependency>

配置docket

@Configuration
@EnableSwagger2
public class SwaggerConfig {/*** SpringBoot默认已经将classpath:/META-INF/resources/和classpath:/META-INF/resources/webjars/映射* 所以该方法不需要重写,如果在SpringMVC中,可能需要重写定义(我没有尝试)* 重写该方法需要 extends WebMvcConfigurerAdapter*/
//    @Override
//    public void addResourceHandlers(ResourceHandlerRegistry registry) {//        registry.addResourceHandler("swagger-ui.html")
//                .addResourceLocations("classpath:/META-INF/resources/");
//
//        registry.addResourceHandler("/webjars/**")
//                .addResourceLocations("classpath:/META-INF/resources/webjars/");
//    }@Beanpublic Docket createRestApi() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).select().apis(RequestHandlerSelectors.basePackage("com.wuwii")).paths(PathSelectors.any()).build();}private ApiInfo apiInfo() {return new ApiInfoBuilder().title("Spring Boot中使用Swagger2构建RESTful APIs").description("rest api 文档构建利器").termsOfServiceUrl("https://blog.wuwii.com/").contact("KronChan").version("1.0").build();}
}
builder说明

根据网上一位前辈的文章:

@Beanpublic Docket petApi() {return new Docket(DocumentationType.SWAGGER_2).select() //1.apis(RequestHandlerSelectors.any()).paths(PathSelectors.any()).build().pathMapping("/") //2.directModelSubstitute(LocalDate.class, //3String.class).genericModelSubstitutes(ResponseEntity.class) //4.alternateTypeRules( //5newRule(typeResolver.resolve(DeferredResult.class,typeResolver.resolve(ResponseEntity.class, WildcardType.class)),typeResolver.resolve(WildcardType.class))).useDefaultResponseMessages(false) //6.globalResponseMessage(RequestMethod.GET, //7newArrayList(new ResponseMessageBuilder().code(500).message("500 message").responseModel(new ModelRef("Error")).build())).securitySchemes(newArrayList(apiKey())) //8.securityContexts(newArrayList(securityContext())) //9;}

方法说明:

  1. 定义了需要生成API文档的endpoint,api()方法可以通过RequestHandlerSelectors的各种选择器来选择,比如说选择所有注解了@RsestController的类中的所有API e.g. .apis(RequestHandlerSelectors.withClassAnnotation(RestController.class))path()方法可以通过PathSelectors的来匹配路径,提供了regex匹配或者ant匹配
  2. 定义了API的根路径
  3. 输出模型定义时的替换,比如遇到所有LocalDate的字段时,输出成String
  4. 遇到对应泛型类型的外围类,直接解析成泛型类型,比如说ResponseEntity,应该直接输出成类型T
  5. 提供了自定义性更强的针对泛型的处理,示例中的代码的意思是将类型DeferredResult直接解析成类型T
  6. 是否使用默认的ResponseMessage, 框架默认定义了一些针对各个HTTP方法时各种不同响应值对应的message
  7. 全局的定义ResponseMessage,示例代码定义GET方法的500错误的消息以及错误模型。注意这里GET方法的所有ResponseMessage都会被这里的定义覆盖
  8. 定义API支持的SecurityScheme,指的是认证方式,支持OAuthAPIkey。 P.S. 要让swagger-ui的oauth正常工作,需要定义个SecurityConfiguration的Bean
  9. 定义具体上下文路径对应的认证方式
  10. 还有一些接口可以定义API的名称等一些基本信息,定义API支持的数据格式等等。

接口上添加文档

@RestController
@Api(description = "这是一个控制器的描述 ")
public class PetController {/*** logger*/private static final Logger LOGGER = LoggerFactory.getLogger(PetController.class);private String no;private String kind;private String name;@ApiOperation(value="测试接口", notes="测试接口描述")@ApiImplicitParams({@ApiImplicitParam(name = "id", value = "用户ID", required = true, dataType = "Long", paramType = "path"),@ApiImplicitParam(name = "pet", value = "宠物", required = true, dataType = "PetController")})@ApiResponses({@ApiResponse(code = 200, message = "请求完成"),@ApiResponse(code = 400, message = "请求参数错误")})@RequestMapping(path = "/index/{id}", method = RequestMethod.PUT)public PetController index1(@PathVariable("id") String id, @RequestBody PetController pet) {return pet;}//…… get  / set
常用的注解说明

查看API文档

启动Spring Boot程序,访问:http://host:port/swagger-ui.html
。就能看到RESTful API的页面。打开我们的测试接口的API ,可以查看这个接口的描述,以及参数等信息:

点击上图中右侧的Model Schema(黄色区域:它指明了这个requestBody的数据结构),此时pet中就有了pet对象的模板,修改上测试数据,点击下方Try it out!按钮,即可完成了一次请求调用!

调用完后,我们可以查看接口的返回信息:

参考文章

  • Spring Boot中使用Swagger2构建强大的RESTful API文档档
  • spring-boot-swagger2 使用手册
  • 使用springfox生成springmvc项目的swagger的文档

例外补充点

验证码

我使用的是 com.github.axet.kaptcha 的验证码
虽然按照别人的方法使用 HttpServletResponse 输出流,这种是暴露 Servlet 的接口。但是发现了一个问题了,在 swagger 的获取验证码接上测试的时候不能得到验证码图片,但是在 img 标签中是没问题,发现 swagger 还是把我的返回结果作为 json 处理。所以我还是想到使用下载中二进制流的方法,将 BufferedImage 转换成二进制流数组,总算是解决。

上最后解决的办法:

/*** 获取验证码*/@GetMapping(value = "/captcha.jpg", produces = MediaType.IMAGE_JPEG_VALUE)public ResponseEntity<byte[]> captcha()throws IOException {//生成文字验证码String text = producer.createText();//生成图片验证码BufferedImage image = producer.createImage(text);ByteArrayOutputStream out = new ByteArrayOutputStream();ImageIO.write(image, "jpg", out);// 文字验证码保存到 shiro sessionShiroUtils.setSessionAttribute(Constants.KAPTCHA_SESSION_KEY, text);HttpHeaders headers = new HttpHeaders();headers.setCacheControl("no-store, no-cache");return ResponseEntity.status(HttpStatus.OK).headers(headers).body(out.toByteArray());

我是采用 ImageIO 工具类的将 BufferedImage 转换成 输出流,从输出流中获取二进制流数组。

ImageIO.write(BufferedImage image,String format,OutputStream out)

再补充一个 将二进制流数组 byte[] 转换成 BufferedImage

// 将二进制流数组转换成输入流
ByteArrayInputStream in = new ByteArrayInputStream(byte[] byets);
// 读取输入流
BufferedImage image = ImageIO.read(InputStream in);

swagger 上是这样的了:

配置不同环境中是否启动

在不同环境种配置是否启用规则:

swagger:enable: true  # or false

在 swagger 配置类中加入

    /*** 启用*/@Value("${swagger.enable}")private boolean enable;…… set / get

配置 Docket 中

    @Beanpublic Docket createRestApi() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo())// 加入 enable.enable(enable).select().apis(RequestHandlerSelectors.basePackage("com.wuwii")).paths(PathSelectors.any()).build();}

SpringMVC 中配置 Swagger2

Swagger 的配置文件:

@Configuration
@EnableSwagger2
@EnableWebMvc
@ComponentScan("com.devframe.controller")
public class SwaggerConfig extends WebMvcConfigurerAdapter {/*** 静态文件过滤* @param registry*/@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {registry.addResourceHandler("swagger-ui.html").addResourceLocations("classpath:/META-INF/resources/");registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");}@Beanpublic Docket createRestApi() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).select().apis(RequestHandlerSelectors.basePackage("com.devframe")).paths(PathSelectors.any()).build();}private ApiInfo apiInfo() {return new ApiInfoBuilder().title("RESTful APIs").description("rest api 文档构建利器").termsOfServiceUrl("https://blog.wuwii.com/").version("1.0").build();}}

额外需要在 web.xml 配置:

    <!-- Springmvc前端控制器扫描路径增加“/v2/api-docs”,用于扫描Swagger的 /v2/api-docs,否则 /v2/api-docs无法生效。--><servlet-mapping><servlet-name>dispatcher</servlet-name><url-pattern>/v2/api-docs</url-pattern></servlet-mapping>

单元测试中会出现的错误

发现加入 Swagger 后,以前的单元测试再运行的时候,会抛出一个异常,参考 How to run integration tests with spring and springfox?

Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'documentationPluginsBootstrapper'
defined in URL [jar:file:/C:/Users/test/.m2/repository/io/springfox/springfox-spring-web/2.0.0-SNAPSHOT/springfox-spring-web-2.0.0-
SNAPSHOT.jar!/springfox/documentation/spring/web/plugins/DocumentationPluginsBootstrapper.class]: Unsatisfied dependency expressed through
constructor argument with index 1 of type [java.util.List]: : No qualifying bean of type
[org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping] found for dependency [collection of
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping]: expected at least 1 bean which qualifies as autowire
candidate for this dependency. Dependency annotations: {}; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException:
No qualifying bean of type [org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping] found for dependency
[collection of org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping]: expected at least 1 bean which qualifies as
autowire candidate for this dependency. Dependency annotations: {}1

解决,单元测试上加入 @EnableWebMvc

学习Spring Boot:(六) 集成Swagger2相关推荐

  1. quartz集成到springboot_一个用来学习 spring boot 的开源项目,总共包含 57 个集成demo...

    spring-boot-demo 一个用来学习 spring boot 的项目,总共包含 57 个集成demo. 包含集成 actuator(监控).admin(可视化监控).logback(日志). ...

  2. SpringBoot2.x系列教程(六十六)Spring Boot快速集成RocketMQ实战教程

    前言 RocketMQ是目前主流的消息中间件之一,并且自身就支持分布式功能.最初由阿里巴巴团队开发,并且经历过双十一等海量消息场景的考验,后捐赠给Apache开源基金会,这也是为什么我们经常听说Roc ...

  3. 学习Spring Boot:(二十六)使用 RabbitMQ 消息队列

    前言 前面学习了 RabbitMQ 基础,现在主要记录下学习 Spring Boot 整合 RabbitMQ ,调用它的 API ,以及中间使用的相关功能的记录. 相关的可以去[我的博客/Rabbit ...

  4. Spring Boot中使用Swagger2构建RESTful APIs

    关于 Swagger Swagger能成为最受欢迎的REST APIs文档生成工具之一,有以下几个原因: Swagger 可以生成一个具有互动性的API控制台,开发者可以用来快速学习和尝试API. S ...

  5. springboot 拦截器 日志_跟武哥一起学习Spring Boot,一份全面详细的学习教程

    SpringBoot现在已经是企业开发项目的标准框架,至少新项目一般都会使用SpringBoot框架,发现有一个SpringBoot的笔记写的非常好,非常全面, 也非常认真,即使你对SpringBoo ...

  6. Spring Boot 工程集成全局唯一ID生成器 Vesta

    2019独角兽企业重金招聘Python工程师标准>>> 本文内容脑图如下: 文章共 760字,阅读大约需要 2分钟 ! 概 述 在前一篇文章 <Spring Boot工程集成全 ...

  7. 学习Spring Boot:(一)入门

    微服务 微服务其实是服务化思路的一种最佳实践方向,遵循SOA(面向服务的架构)的思路,各个企业在服务化治理上面的道路已经走得很远了,整个软件交付链上各个环节的基础设施逐渐成熟了,微服务就诞生了. 微服 ...

  8. 2022 年学习 Spring Boot 开发的最佳书籍

    在我们之前的文章中,我们查看了学习 Java 编程的必读书籍我们在其中探索了一些您可以利用的资源来加快 Java 开发的速度.在此基础上,在用 vanilla Java 编写一段时间后,您将意识到组织 ...

  9. 在spring boot中集成Swagger

    Swagger 在spring boot中集成Swagger 新建一个swagger项目 maven依赖 <!-- https://mvnrepository.com/artifact/io.s ...

最新文章

  1. Mysql大表的优化
  2. 最懂中文的H5前端框架amazeUI
  3. Redis-Session无状态会话技术
  4. HDU 2502 月之数(简单递推)
  5. Python 字典删除元素clear、pop、popitem
  6. sku设计mysql_基于spring boot拥有完整sku和下单流程的完全商城
  7. 2019传智python就业班视频_2019黑马JAVAEE57期基础班就业班(全套)
  8. CSUOJ 1215: 稳定排序
  9. linux操作系统-SSH原理介绍与免密运用与远程命令操作
  10. 博客园鼠标点击烟花特效
  11. Web SQL Database
  12. 2.3.4nbsp;《孙子兵法》的基本原则
  13. SpringBoot个人博客项目——相册模块开发(一)
  14. Android RecyclerView使用ItemDecoration刻画时间线/时间轴/时光轴timeline
  15. 3D (Input) Sparse Convolution
  16. Maven2的相关知识[zt]
  17. ospf-NBMA基本配置
  18. mysql超链接_PHP MYSQL超链接和表单
  19. intel至强服务器芯片制程,64核自研芯片性能提升7倍,追平英特尔至强
  20. AndroidStudio如何打包生成realease版本的arr包,并上传到Nexus搭建的maven仓库,供项目远程依赖(二)

热门文章

  1. 编译安装mysql5.5.39_编译安装MySQL5.5
  2. linux虚拟机怎么看var文件,一种获取Linux虚拟机内部日志的方法
  3. mysql 输出参数 sql语句_MySQL: 详细的sql语句
  4. 链接服务器访问接口返回了消息没有活动事务,因为链接服务器 SQLEHR 的 OLE DB 访问接口 SQLNCLI10 无法启动分布式事务。...
  5. 本地连接和音量图标显示
  6. python设置本机IP地址、子网掩码、DNS,获取本机IP地址、子网掩码、DNS、MAC
  7. C++PrimerPlus学习——第十三章编程练习
  8. mysql 字符串函数大全_mysql 字符串函数大全
  9. java循环队列_Java 循环队列的实现
  10. mysql 时间差_后端从mysql取值返回0时区时间数据的问题