OpenAPIv3
协议主要使用在规范路由中,阅读接口文档协议介绍之前,请先了解一下规范路由: 路由注册-规范路由
一、 OpenAPIv3
详细的 OpenAPIv3
协议介绍请参考: https://swagger.io/specification/
二、 g.Meta
元数据
接口的元数据信息可以通过为输入结构体embedded
方式嵌入g.Meta
结构,并通过g.Meta
的属性标签方式来实现。
关于元数据组件的介绍,详情请参考章节: 元数据-gmeta
三、常用协议标签
输入输出结构体中的属性的标签完整支持 OpenAPIv3
协议,因此只要增加了对应的协议标签,那么生成的 OpenAPIv3
接口信息中将会自动包含该属性。
大部分的标签属性已经被 Server
组件自动生成,开发者需要手动设置的标签不多。
1、基础标签
常见的基础标签包括:
常见标签 | 说明 | 备注 |
---|---|---|
path | 结合注册时的前缀共同构成接口URI路径 | 用于 g.Meta 标识接口元数据 |
tags | 接口所属的标签,用于接口分类 | 用于 g.Meta 标识接口元数据 |
method | 接口的请求方式: ALL/GET/PUT/POST/DELETE...(不区分大小写) 。支持组合方式如GET,POST ,多种方式使用, 分隔。 | 用于 g.Meta 标识接口元数据 |
deprecated | 标记该接口废弃 | 用于 g.Meta 标识接口元数据 |
summary | 接口/参数概要描述 | 缩写 sm |
description | 接口/参数详细描述 | 缩写 dc |
in | 参数的提交方式 | header/path/query/cookie |
default | 参数的默认值 | 缩写 d |
mime | 接口的 MIME 类型,例如 multipart/form-data 一般是全局设置,默认为 application/json 。 | 用于 g.Meta 标识接口元数据 |
type | 参数的类型,一般不需要设置,特殊参数需要手动设置,例如 file | 仅用于参数属性 |
更多标签请参考标准的 OpenAPIv3
协议: https://swagger.io/specification/
2、扩展标签
在 OpenAPI
规范里面,所有名称以 x-
开头的标签是开发者可自定义的扩展标签。扩展标签可以在任意的接口、属性中以 Golang struct tag
的形式定义,在接口文档生成时,将会作为独立的字段返回。例如:
package main
import (
"context"
"github.com/gogf/gf/v2/errors/gcode"
"github.com/gogf/gf/v2/errors/gerror"
"github.com/gogf/gf/v2/frame/g"
"github.com/gogf/gf/v2/net/ghttp"
)
type GetListReq struct {
g.Meta `path:"/user" tags:"User" method:"get" x-group:"User/Info" summary:"Get user list with basic info."`
Page int `dc:"Page number" d:"1" x-sort:"1"`
Size int `dc:"Size for per page." d:"10" x-sort:"2"`
}
type GetListRes struct{}
type Controller struct{}
func (c *Controller) GetList(ctx context.Context, req *GetListReq) (res *GetListRes, err error) {
return nil, gerror.NewCode(gcode.CodeNotImplemented)
}
func main() {
s := g.Server()
s.Group("/", func(group *ghttp.RouterGroup) {
group.Bind(new(Controller))
})
s.SetOpenApiPath("/api.json")
s.SetSwaggerPath("/swagger")
s.SetPort(8199)
s.Run()
}
执行后,访问地址 http://127.0.0.1:8199/swagger 可以查看 swagger ui
,访问 http://127.0.0.1:8199/api.json 可以查看对应的 OpenAPIv3
接口文档。其中生成的 OpenAPIv3
接口文档如下:
{
"openapi": "3.0.0",
"components": {
"schemas": {
"main.GetListReq": {
"properties": {
"Page": {
"default": 1,
"description": "Page number",
"format": "int",
"properties": {},
"type": "integer",
"x-sort": "1"
},
"Size": {
"default": 10,
"description": "Size for per page.",
"format": "int",
"properties": {},
"type": "integer",
"x-sort": "2"
}
},
"type": "object",
"x-group": "User/Info"
},
"main.GetListRes": {
"properties": {},
"type": "object"
}
}
},
"info": {
"title": "",
"version": ""
},
"paths": {
"/user": {
"get": {
"parameters": [
{
"description": "Page number",
"in": "query",
"name": "Page",
"schema": {
"default": 1,
"description": "Page number",
"format": "int",
"properties": {},
"type": "integer",
"x-sort": "1"
},
"x-sort": "1"
},
{
"description": "Size for per page.",
"in": "query",
"name": "Size",
"schema": {
"default": 10,
"description": "Size for per page.",
"format": "int",
"properties": {},
"type": "integer",
"x-sort": "2"
},
"x-sort": "2"
}
],
"responses": {
"200": {
"content": {
"application/json": {
"schema": {
"$ref": "#/components/schemas/main.GetListRes"
}
}
},
"description": ""
}
},
"summary": "Get user list with basic info.",
"tags": [
"User"
],
"x-group": "User/Info"
},
"x-group": "User/Info"
}
}
}
可以看到,扩展标签已经生成到了接口文档中。
四、数据校验规则自动转换
GoFrame
的goai
组件能够自动将结构体字段的数据校验规则(validation rules
)转换为OpenAPIv3
中对应的校验数据结构,使接口文档更加准确和完整。这意味着您只需在结构体字段中定义常规的数据校验规则,goai
组件就会自动将其转换为OpenAPIv3
规范中的对应约束。
1. 支持的校验规则
以下是goai
组件自动识别并转换的校验规则:
校验规则 | 说明 | OpenAPIv3对应结构 |
---|---|---|
required | 必填字段 | required: true |
min:{数值} | 最小值限制 | minimum: {数值} |
max:{数值} | 最大值限制 | maximum: {数值} |
length:{最小长度},{最大长度} | 字符串长度范围 | minLength: {最小长度}, maxLength: {最大长度} |
min-length:{长度} | 字符串最小长度 | minLength: {长度} |
max-length:{长度} | 字符串最大长度 | maxLength: {长度} |
between:{最小值},{最大值} | 数值范围 | minimum: {最小值}, maximum: {最大值} |
in:{值1},{值2},... | 枚举值限制 | enum: [{值1}, {值2}, ...] |
2. 使用示例
type CreateUserReq struct {
g.Meta `path:"/user" method:"post" summary:"创建用户"`
Name string `v:"required|length:2,30" dc:"用户名"`
Age int `v:"required|between:18,60" dc:"年龄"`
Email string `v:"required|email" dc:"电子邮箱"`
Score int `v:"min:0|max:100" dc:"分数"`
Type string `v:"in:admin,user,guest" dc:"用户类型"`
}
生成的OpenAPIv3
文档将包含以下校验信息:
Name
字段:required: true, minLength: 2, maxLength: 30
Age
字段:required: true, minimum: 18, maximum: 60
Email
字段:required: true
Score
字段:minimum: 0, maximum: 100
Type
字段:enum: ["admin", "user", "guest"]
3. 注意事项
- 数值类型(如
int
、float
等)的字段才会转换min
、max
和between
规则为minimum
和maximum
- 字符串类型的字段会转换
min-length
、max-length
和length
规则为minLength
和maxLength
required
规则会转换为参数的required: true
属性in
规则会转换为enum
数组,如果所有值都是数字,则会转换为数字类型的枚举,否则为字符串类型
五、扩展 OpenAPIv3
信息
核心的接口信息已经自动生成,如果开发者想要更进一步完善接口信息,可以通过 s.GetOpenApi()
接口获取到 OpenAPIv3
的结构体对象,并手动填充对应的属性内容即可。我们来看一个示例,在该示例中,我们设计每个接口外层公共的数据结构:
我们可以发现通过通用的 OpenAPIv3
对象我们可以自定义修改其内容,并且根据它生成其他各种自定义类型的接口文档。
五、添加api.json(swagger)
自定义鉴权
对于需要进行api
文档鉴权的情况,可以使用 ghttp.BindHookHandler
方法对 s.GetOpenApiPath()
路由绑定前置方法进行鉴权,示例如下:
func main() {
s := g.Server()
// if api.json requires authentication, add openApiBasicAuth handler
s.BindHookHandler(s.GetOpenApiPath(), ghttp.HookBeforeServe, openApiBasicAuth)
s.Run()
}
func openApiBasicAuth(r *ghttp.Request) {
if !r.BasicAuth("OpenApiAuthUserName", "OpenApiAuthPass", "Restricted") {
r.ExitAll()
return
}
}
详细介绍请参考代码示例文档:swagger-auth