SpringBoot - 接口文档之Smart-Doc
# SpringBoot - 接口文档之Smart-Doc
# swagger vs smart-doc
首先我们先看一下Swagger组件目前存在的主要问题:
Swagger的代码侵入性比较强
这个很容易理解,要让Swagger生成接口文档必须要给方法或字段添加对应的注解,是存在代码侵入的。
原生swagger不支持接口的参数分组
对于有做参数分组的接口,原生的Swagger并未支持,虽然我们通过扩展其组件可以让其支持参数分组,但是毕竟要开发,而且还未支持最新的Swagger3版本。
那作为对比,smart-doc
是基于接口源码分析来生成接口文档,完全做到零注解侵入,你只需要按照java标准注释的写,smart-doc就能帮你生成一个简易明了的markdown 或是一个像GitBook样式的静态html文档。官方地址:https://gitee.com/smart-doc-team/smart-doc
简单罗列一下smart-doc的优点:
- 零注解、零学习成本、只需要写标准java注释即可生成文档。
- 基于源代码接口定义自动推导,强大的返回结构推导。
- 支持Spring MVC,Spring Boot,Spring Boot Web Flux(controller书写方式)。
- 支持Callable,Future,CompletableFuture等异步接口返回的推导。
- 支持JavaBean上的JSR303参数校验规范,支持参数分组。
- 对一些常用字段定义能够生成有效的模拟值。
- ...
接下来我们来看看SpringBoot中如何集成smart-doc。
# SpringBoot集成 smart-doc
smart-doc支持多种方式生成接口文档:maven插件、gradle插件、单元测试(不推荐),这里用的是基于maven插件生成,步骤如下:
- 引入依赖
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<groupId>com.github.shalousun</groupId>
<artifactId>smart-doc-maven-plugin</artifactId>
<version>2.6.7</version>
<configuration>
<!--指定生成文档的使用的配置文件,配置文件放在自己的项目中-->
<configFile>./src/main/resources/smart-doc.json</configFile>
<!--指定项目名称,推荐使用动态参数,例如${project.description}-->
<!--如果smart-doc.json中和此处都未设置projectName,2.3.4开始,插件自动采用pom中的projectName作为设置-->
<!--<projectName>${project.description}</projectName>-->
<!--smart-doc实现自动分析依赖树加载第三方依赖的源码,如果一些框架依赖库加载不到导致报错,这时请使用excludes排除掉-->
<excludes>
<!--格式为:groupId:artifactId;参考如下-->
<!--也可以支持正则式如:com.alibaba:.* -->
<exclude>com.alibaba:fastjson</exclude>
</excludes>
<!--includes配置用于配置加载外部依赖源码,配置后插件会按照配置项加载外部源代码而不是自动加载所有,因此使用时需要注意-->
<!--smart-doc能自动分析依赖树加载所有依赖源码,原则上会影响文档构建效率,因此你可以使用includes来让插件加载你配置的组件-->
<includes>
<!--格式为:groupId:artifactId;参考如下-->
<!--也可以支持正则式如:com.alibaba:.* -->
<include>com.alibaba:fastjson</include>
</includes>
</configuration>
<executions>
<execution>
<!--如果不需要在执行编译时启动smart-doc,则将phase注释掉-->
<phase>compile</phase>
<goals>
<!--smart-doc提供了html、openapi、markdown等goal,可按需配置-->
<goal>html</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
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
重点在 configFile
中指定smart-doc配置文件 smart-doc.json
- 新建配置文件smart-doc.json
{
"outPath": "src/main/resources/static/doc",
"allInOne": true,
"allInOneDocFileName":"index.html",
"projectName": "springboot-smart-doc"
}
2
3
4
5
6
指定smart-doc生成的文档路径,其他配置项可以参考官方wiki。
通过执行maven 命令生成对应的接口文档(html)
mvn -Dfile.encoding=UTF-8 smart-doc:html
1当然也可以通过idea中的maven插件生成。
smart-doc 还支持生成如下类型的文档:
//生成markdown mvn -Dfile.encoding=UTF-8 smart-doc:markdown //生成adoc mvn -Dfile.encoding=UTF-8 smart-doc:adoc //生成postman json数据 mvn -Dfile.encoding=UTF-8 smart-doc:postman // 生成 Open Api 3.0+, Since smart-doc-maven-plugin 1.1.5 mvn -Dfile.encoding=UTF-8 smart-doc:openapi
1
2
3
4
5
6
7
8访问接口文档
生成接口文档后打开html查看,效果如下:
别急,刚刚只是体验了smart-doc的基础功能,接下来我们通过丰富smart-doc的配置文件内容来增强其功能。
# 功能增强
# 1. 开启调试
一个优秀的接口文档工具调试功能必不能少,smart-doc支持在线调试功能,只需要加入如下几个配置项:
{
"serverUrl": "http://localhost:8096", -- 服务器地址
"allInOne": true, -- 是否将文档合并到一个文件中,一般推荐为true
"outPath": "src/main/resources/static/doc", -- 指定文档的输出路径
"createDebugPage": true, -- 开启测试
"allInOneDocFileName":"index.html", -- 自定义文档名称
"projectName": "springboot-smart-doc" -- 项目名称
}
2
3
4
5
6
7
8
9
通过"createDebugPage": true 开启debug功能,在让生成smart-doc生成文档时直接放入到 static/doc/
下,这样可以直接启动程序访问页面 http://localhost:8080/doc/index.html
进行开发调试。
有的开发人员直接在idea中使用【Open In Browser】打开smart-doc生成的debug页面,如果非要这做,前端js请求后台接口时就出现了跨域。因此你需要在后端配置跨域。
# 2. 通用响应体
返回统一的数据结构,我们需要让其在接口文档中也有此功能,在配置文件追加配置内容:
{
"responseBodyAdvice":{ -- 通用响应体
"className":"cn.jtoss.springbootsmartdoc.response.ResultResponse"
}
}
2
3
4
5
# 3. 自定义Header
在前后端分离项目中我们一般需要在请求接口时设置一个请求头,如token,Authorization等...后端根据请求头判断是否为系统合法用户,目前smart-doc也对其提供了支持。
在smart-doc配置文件 smart-doc.json
中继续追加如下配置内容:
"requestHeaders": [ //设置请求头,没有需求可以不设置
{
"name": "Authorization",//请求头名称
"type": "string",//请求头类型
"desc": "自定义请求头 - token",//请求头描述信息
"value":"123456",//不设置默认null
"required": false,//是否必须
"since": "-",//什么版本添加的改请求头
"pathPatterns": "/smart/say",//只有以/smart/say 开头的url才会有此请求头
"excludePathPatterns":"/smart/add,/smart/edit" // url=/app/page/将不会有该请求头
}
]
2
3
4
5
6
7
8
9
10
11
12
效果如下:
# 4. 参数分组
演示一下smart-doc对于参数分组的支持
新增操作时,name、level为必填项,email、sex为非必填。
编辑操作时,id,appId,level为必填项,email、sex为非必填。
通过上面的效果可以看出smart-doc对于参数分组是完全支持的。
# 完整配置
附上完整配置说明:
{
"serverUrl": "http://127.0.0.1:8096", //服务器地址,非必须。导出postman建议设置成http://{{server}}方便直接在postman直接设置环境变量
"pathPrefix": "", //设置path前缀,非必须。如配置Servlet ContextPath 。@since 2.2.3
"isStrict": false, //是否开启严格模式
"allInOne": true, //是否将文档合并到一个文件中,一般推荐为true
"outPath": "src/main/resources/static/doc", //指定文档的输出路径
"coverOld": true, //是否覆盖旧的文件,主要用于markdown文件覆盖
"createDebugPage": true,//@since 2.0.0 smart-doc支持创建可以测试的html页面,仅在AllInOne模式中起作用。
"packageFilters": "",//controller包过滤,多个包用英文逗号隔开,2.2.2开始需要采用正则:com.test.controller.*
"md5EncryptedHtmlName": false,//只有每个controller生成一个html文件时才使用
"style":"xt256", //基于highlight.js的代码高设置,可选值很多可查看码云wiki,喜欢配色统一简洁的同学可以不设置
"projectName": "pringboot-smart-doc",//配置自己的项目名称,不设置则插件自动获取pom中的projectName
"skipTransientField": true,//目前未实现
"sortByTitle":false,//接口标题排序,默认为false,@since 1.8.7版本开始
"showAuthor":true,//是否显示接口作者名称,默认是true,不想显示可关闭
"requestFieldToUnderline":true,//自动将驼峰入参字段在文档中转为下划线格式,//@since 1.8.7版本开始
"responseFieldToUnderline":true,//自动将驼峰入参字段在文档中转为下划线格式,//@since 1.8.7版本开始
"inlineEnum":true,//设置为true会将枚举详情展示到参数表中,默认关闭,//@since 1.8.8版本开始
"recursionLimit":7,//设置允许递归执行的次数用于避免一些对象解析卡主,默认是7,正常为3次以内,//@since 1.8.8版本开始
"allInOneDocFileName":"index.html",//自定义设置输出文档名称, @since 1.9.0
"requestExample":"true",//是否将请求示例展示在文档中,默认true,@since 1.9.0
"responseExample":"true",//是否将响应示例展示在文档中,默认为true,@since 1.9.0
"ignoreRequestParams":[ //忽略请求参数对象,把不想生成文档的参数对象屏蔽掉,@since 1.9.2
"org.springframework.ui.ModelMap"
],
"dataDictionaries": [{ //配置数据字典,没有需求可以不设置
"title": "http状态码字典", //数据字典的名称
"enumClassName": "...", //数据字典枚举类名称
"codeField": "responseCode",//数据字典字典码对应的字段名称
"descField": "description"//数据字典对象的描述信息字典
}],
"errorCodeDictionaries": [{ //错误码列表,没有需求可以不设置
"title": "title",
"enumClassName": "...", //错误码枚举类
"codeField": "responseCode",//错误码的code码字段名称
"descField": "description"//错误码的描述信息对应的字段名
}],
"revisionLogs": [{ //文档变更记录,非必须
"version": "1.1", //文档版本号
"revisionTime": "2022-09-01 22:12:01", //文档修订时间
"status": "update", //变更操作状态,一般为:创建、更新等
"author": "jason", //文档变更作者
"remarks": "init user api" //变更描述
},{ //文档变更记录,非必须
"version": "1.2", //文档版本号
"revisionTime": "2022-09-01 22:12:02", //文档修订时间
"status": "update", //变更操作状态,一般为:创建、更新等
"author": "jason", //文档变更作者
"remarks": "add address api" //变更描述
}
],
"customResponseFields": [{ //自定义添加字段和注释,一般用户处理第三方jar包库,非必须
"name": "code",//覆盖响应码字段
"desc": "响应代码",//覆盖响应码的字段注释
"ownerClassName": "org.springframework.data.domain.Pageable", //指定你要添加注释的类名
"ignore":true, //设置true会被自动忽略掉不会出现在文档中
"value": "00000"//设置响应码的值
}],
"requestHeaders": [{ //设置请求头,没有需求可以不设置
"name": "token",//请求头名称
"type": "string",//请求头类型
"desc": "desc",//请求头描述信息
"value":"token请求头的值",//不设置默认null
"required": false,//是否必须
"since": "-",//什么版本添加的改请求头
"pathPatterns": "/valid/add",//请看https://gitee.com/smart-doc-team/smart-doc/wikis/请求头高级配置?sort_id=4178978
"excludePathPatterns":"/valid/edit"//请看https://gitee.com/smart-doc-team/smart-doc/wikis/请求头高级配置?sort_id=4178978
}],
"requestParams": [ //设置公共参数,没有需求可以不设置
{
"name": "configPathParam",//请求名称
"type": "string",//请求类型
"desc": "desc",//请求描述信息
"paramIn": "path", // 参数所在位置 header-请求头, path-路径参数, query-参数
"value":"testPath",//不设置默认null
"required": false,//是否必须
"since": "2.2.3",//什么版本添加的该请求
"pathPatterns": "/app/test/**",//请看https://gitee.com/smart-doc-team/smart-doc/wikis/请求高级配置?sort_id=4178978
"excludePathPatterns":"/app/page/**"//请看https://gitee.com/smart-doc-team/smart-doc/wikis/请求高级配置?sort_id=4178978
}],
"responseBodyAdvice":{ //自smart-doc 1.9.8起,非必须项,ResponseBodyAdvice统一返回设置(不要随便配置根据项目的技术来配置),可用ignoreResponseBodyAdvice tag来忽略
"className":"cn.jtoss.springbootsmartdoc.response.ResultResponse" //通用响应体
}
}
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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
# 进一步理解
结合smart-doc官方文档,我们通过几个问题进一步理解smart-doc。主要内容来源于官方文档 (opens new window)。
我们知道注释的信息是有限的,swagger技术栈的方式通过定义注解来约束并拓展文档中的内容,那么smart-doc如何从注释拓展文档内容呢?
一方面smart-doc
的实现初衷是通过使用javadoc
文档注释来去除注解式的侵入,因此smart-doc
每增加一个功能首先都是去考虑javadoc
原生的tag
,
下面对smart-doc
使用的一些javadoc
的注释tag
做介绍。
tag名称 | 使用描述 |
---|---|
@param | 对于在Spring Boot 接口层,对于简单类型的参数必须在使用@param 时写上注释描述,对于Entity 类型smart-doc 则不会检查 |
@deprecated | 可以在注释中用于标记接口已经废弃,作用同@Deprecated 注解 |
@apiNote | @apiNote 是JAVA 新增的文档tag ,smart-doc 使用@apiNote 的注释作为方法的详细描述,因此可以使用@apiNote 来写一段长注释。如果一个方法不写 @apiNote 注释说明,smart-doc 直接使用方法默认注释填充 |
另一方面,原生的tag是不够的,所以smart-doc
又通过自定义tag来支持更多功能的拓展
tag名称 | 描述 |
---|---|
@ignore | @ignore tag 用于过滤请求参数对象上的某个字段,设置后smart-doc 不输出改字段到请求参数列表中。关于响应字段忽略的请看【忽略响应字段】 (opens new window) 如果@ignore 加到方法上,则接口方法不会输出到文档。从1.8.4 开始@ignore 支持添加到Controller 上进行忽略不想生成文档的接口类。@ignore 也可以用于方法上忽略某个请求参数。 |
@required | 如果你没有使用JSR303 参数验证规范实现的方式来标注字段,就可以使用@required 去标注请求参数对象的字段,标注smart-doc 在输出参数列表时会设置为true 。 |
@mock | 从smart-doc 1.8.0 开始,@mock tag 用于在对象基本类型字段设置自定义文档展示值。设置值后smart-doc 不再帮你生成随机值。方便可以通过smart-doc 直接输出交付文档。 |
@dubbo | 从smart-doc 1.8.7 开始,@dubbo tag 用于在Dubbo 的API 接口类上添加让smart-doc 可以扫描到Dubbo RPC 的接口生成文档。 |
@restApi | 从smart-doc 1.8.8 开始,@restApi tag 用于支持smart-doc 去扫描Spring Cloud Feign 的定义接口生成文档。 |
@order | 从smart-doc 1.9.4 开始,@order tag 用于设置Controller 接口或者API 入口的自定义排序序号,@order 1 就表示设置序号为1 。 |
@ignoreResponseBodyAdvice | 从smart-doc 1.9.8 开始,@ignoreResponseBodyAdvice tag 用于忽略ResponseBodyAdvice 设置的包装类。 |
@download | 从smart-doc 2.0.1 开始,@download tag 用于标注在Controller 的文件下载方法上,生成debug 页面时可实现文件下载测试。并且支持下载文件带请求头参数测试。 |
@page | 从smart-doc 2.0.2 开始,@page tag 用于标注在Controller 的方法上表示该方法用来渲染返回一个静态页面,生成debug 页面时如果发起测试,测试页面会自动在浏览器开启新标签显示页面。 |
@ignoreParams | 从smart-doc 2.1.0 开始,@ignoreParams tag 用于标注在Controller 方法上忽略掉不想显示在文档中的参数,例如:@ignoreParams id name ,多个参数名用空格隔开 |
@response | 从smart-doc 2.2.0 开始,@response tag 标注在Controller 方法上可以允许用这自己定义返回的json example 。建议只在返回基础类型时使用,如:Result<String> 类型这种泛型是简单原生类型的响应。 |
@tag | @since 2.2.5 , @tag 用于将Controller 方法分类, 可以将不同Contoller 下的方法指定到多个分类下, 同时也可以直接指定Controller 为一个分类或多个分类 |
Maven多模块中使用插件有没有比较好的实践?
在独立的Maven项目中使用smart-doc,当前可以说是如丝般爽滑。但是在Maven的多模块项目中使用smart-doc-maven-plugin时,很多同学就有疑问了, smart-doc插件我到底是放在什么地方合适?是放在Maven的根pom.xml中?还是说各个需要生成API接口文档的模块中呢? 下面就来说说根据不同的项目结构应该怎么放插件。
完全的父子级关系的maven项目:
├─parent
├──common
│ pom.xml
├──web1
│ pom.xml
├──web2
│ pom.xml
└─pom.xml
2
3
4
5
6
7
8
上面的maven结构假设是严格按照父子级来配置的,然后web1和web2都依赖于common, 这种情况下如果跑到web1下或者web2目录下直接执行mvn命令来编译 都是无法完成的。需要在根目录上去执行命令编译命令才能通过,而smart-doc插件会通过类加载器去加载用户配置的一些类,因此是需要调用编译的和执行命令 是一样的。这种情况下建议你建smart-doc-maven-plugin放到根pom.xml中,在web1和web2中放置各自的smart-doc.json配置。 然后通过-pl去指定让smart-doc生成指定 模块的文档。操作命令如下:
# 生成web1模块的api文档
mvn smart-doc:markdown -Dfile.encoding=UTF-8 -pl :web1 -am
# 生成web2模块的api文档
mvn smart-doc:markdown -Dfile.encoding=UTF-8 -pl :web2 -am
2
3
4
如果不是按照严格父子级构建的项目,还是以上面的结构例子来说。common模块放在类parent中,但是common的pom.xml并没有定义parent。 common模块也很少变更,很多公司内部可能就直接把common单独depoly上传到了公司的Nexus仓库中,这种情况下web1和web2虽然依赖于common, 但是web1和web2都可以在web1和web2目录下用命令编译,这种情况下直接将smart-doc-maven-plugin单独放到web1和web2中是可以做构建生成文档的。
# 示例源码
- https://github.com/hengwen/spring-demo/tree/main/springbootsmartdoc
# 参考
- https://pdai.tech/md/spring/springboot/springboot-x-interface-doc-smart.html
- https://juejin.cn/post/7013152753048354823