使用spring-boot-starter-swagger实现API文档化

为什么要实现API文档化?

  1. API文档化有利于前后端分离的开展。随着开发方式全面转成前后端分离,前端和后端的唯一沟通就在API层面。在没有文档化之前,开发人员只能口头的交代并反复理解每个接口参数和返回值,这个过程相当不稳定,变化频繁。而通过API文档化后,可以通过文档的清晰定义,使得前后端人员减少无畏沟通,并在理解上保持一致;

  2. API文档化有利于接口自动化测试。公司马上上马接口自动化测试,对接口参数和返回值要通过Json方式导入自动化测试程序。而用特殊工具生成的API文档,可以直接输出json格式的文档,简化了测试流程,减轻了开发人员的工作量;

  3. API文档化有利于系统文档建设。开发人员不愿意写文档,写注释。而为了以上两个目的必须要求开发人员在编码的同时补充相关的API描述,而通过一个相对稳定成熟和趁手的工具可以大大减轻文档的工作量,做到事半功倍。

Swagger和spring-boot-starter-swagger

Swagger是业界比较流行的实现API文档化的工具。优点有

  1. 通过在项目中引入Swagger,可以使用简单的Annotation,就实现了API文档化;
  2. Swagger提供标准的json或yaml文档,方便做进一步解析,典型应用是接口自动化测试;
  3. 页面可以直接进行测试(try-it-out功能,部分替代Postman);
  4. Swagger还提供类似于github的SwaggerHub,相当于公共的API文档集散地。

Swagger的项目主页:https://swagger.io/

为了将Swagger和公司现有技术框架相结合,在网上发现了这么一个小项目spring-boot-starter-swagger,将公司项目和Swagger的集成又简化了一步,特此向作者表示感谢。
作者在简书上的介绍文章:https://www.jianshu.com/p/91c5a8565a45

主要优点:

  1. 引入足够简单,只需要2步(如果有Shiro就是3步)
  2. 方便的Swagger开关
  3. 支持yaml格式的配置
  4. 支持API的分组显示(在API过多的情况下非常好用)
  5. 支持JSR-303校验注解(@Min, @Max, @Pattern等)
  6. 依然在维护完善,逐步增加更多配置和开关

Swagger引入方法

  1. pom文件引入
        <dependency>
            <groupId>com.spring4all</groupId>
            <artifactId>spring-boot-starter-swagger</artifactId>
            <version>1.5.1.RELEASE</version>
        </dependency>
  1. 主文件加Annotation
@SpringBootApplication
@EnableSwagger2Doc
public calss TestSwaggerApplication{
  1. 增加Shiro的anon设置(防止被登录转向)
client:
  filters:
  filter:
    chain:
      definitions: /swagger-ui.html=anon;/webjars/**=anon;/swagger-resources/**=anon;/v2/api-docs/**=anon;/**=authc;
  1. 配置文件
    配置文件统一放到了项目的application.yml中,稍后计划使用新的文件,并采用引入的方式实现配置文件的分离

4.1 公共信息
定义整个项目的总体信息。

swagger:
  # 公共信息
  enabled: true
  title: swagger-demo
  description: 一个Swagger测试项目
  version: 0.1-SNAPSHOT
  # 许可证及服务条款信息
  license:
  licenseUrl:
  termsOfServiceUrl:
  contact:
    name: Carisma
    url:
    email: carisma.zhao@gmail.com
  base-package: me.learning.swagger

4.2 分组策略
分组策略指的是在整个项目中,将不同类型的API接口分散到不同的页面,方便查看和测试。

# 分组策略
  docket:
    apiForMe:
      title: Myself
      description: 开放给自己的接口
      base-path: /api/myself/**
    apiForOthers:
      title: Others
      description: 作为其他人的接口
      base-path: /api/others/**

4.3 公共参数和通用错误码
像每个接口都需要鉴权这种参数,可以在配置文件中统一定义,这样省去每个接口再写的麻烦,也能兼顾页面的测试。

# 公共参数
  global-operation-parameters[0]:
    name: TOKEN
    description: 鉴权
    modelRef: string
    parameterType: header
    # 公共参数写成requierd, 对于不需要登录的接口随便写一个字符串即可
    required: true
  # 通用返回错误码
  apply-default-response-messages: true
  global-response-message.get[0]:
    code: 401
    message: 401错误
  global-response-message.get[1]:
    code: 500
    message: 后端错误
    modelRef: ERROR

Swagger的效果查看

以上配置完成后,Swagger直接可以生成文档,而不需要单独在每个需要显示的API上添加Annotation,当然下面会讲到如果需要更清晰的显示,还要使用注解,使用注解后,查看方式依然是以下两种。

  1. API文档:在浏览器中输入localhost:9120/swagger-ui.html即可。端口号为Application的启动端口;
  2. Json文件:如果没有分组,则直接使用localhost:9120/swagger-ui.html/v2/api-doc,即显示API的Json格式(包括对model的解析),如果使用了分组,则需要每个分组单独显示,例如localhost:9120/swagger-ui.html/v2/api-doc/?group=apiForMe

Annotation使用详解

要想详细的、切合实际的展示API的内容与要求,则必须在API和Model上加单独的注解,官方JavaDoc虽然不是很详细,但是作为参考也够了(剩下的就是自己试了):
http://docs.swagger.io/swagger-core/current/apidocs/index.html?io/swagger/annotations/Api.html

其中常用的有以下几个,亲测后有些需要详细说明:

@Api
加在需要自定义注解的API类(*Controller)上。这个注解没有什么属性需要说的,其中value比较奇葩,官方说法是隐含的代替tag的作用。(那就用tag不就完了?)为了清晰化接口,使用方式应该是分组>tag>value,即首先用分组分隔,然后使用tag属性,value属性最好就不再使用了。

@ApiOperation
加在API的每个路径映射方法上(即真正的API接口)。value属性是必须的,描述该接口的内容,可以是中文。
response和responseContainer是两个重要的属性。前者表示返回的类,典型的Model.class,注意不是字符串,IDEA是可以解析的(Swagger体系内自动的做成一种Reference);后者可选值为:"List","Set"或者"Map",注意是字符串。

@ApiImplicitParam & @ApiImplicitParams
加在API的参数上,最重要的注解之一。另有一个@ApiParam,是加在函数签名里面,不推荐,这两个是在函数签名的外面,更清晰。后者是前者的集合,比较简单,就不赘述了。前者比较重要的属性有

  1. name:和参数变量名保持一致;
  2. value:对这个参数的简要描述;
  3. required:是否必须;
  4. dataType:数据类型,注意原生类型(包装类也用原生的表示,例如Integer写"int")都是小写,包括"string";
  5. paramType:参数类型,"body"是普通参数,"path"是路径上的那种参数,"query"是url中问号的那种参数;
  6. allowableValues:参数允许的值,这个属性对自动化测试比较重要,具体参考JavaDoc,实际使用中最好写明。需要说明的是,在paramType为body时,range的设置是不生效的(原因未知)。

@ApiModel
以下两个注解都是加在Model里,便于在文档中对自定义类做详细的说明。这个注解加在Model类上。value属性可以提供一个名字,在文档中,会替换原来model的类名(没有试中文是否可行)。同时提供对超类和子类的描述,非字符串,便于生成继承关系(感觉这个功能可以通过代码分析实现)

@ApiModelProperty
加在Model的属性上,对属性做进一步描述和限定。同时这个注解可以加在getXXX方法上,如果XXX属性上也有,则以属性上的注解描述为准。考虑提供方法上加注解,可能是方便描述超类中的这个字段吧(protected类型)。该注解使用方式和@ApiImplicitParam类似。
由于引入了spring-boot-starter-swagger,可以直接支持JSR-303校验类的注解,包括@Pattern、@Max、@Min、@Size等。感觉Pattern是无法用@ApiModelProperty的属性替代的,其他好像都可以,说的不对还请作者指正。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容