go-zero/tools/goctl/api/parser/readme.md
kingxt ee19fb736b
feature: refactor api parse to g4 (#365)
* feature: refactor api parse to g4

* new g4 parser

* add CHANGE_LOG.MD

* refactor

* fix byte bug

* refactor

* optimized

* optimized

* revert

* update readme.md

* update readme.md

* update readme.md

* update readme.md

* remove no need

* fix java gen

* add upgrade

* resolve confilits

Co-authored-by: anqiansong <anqiansong@xiaoheiban.cn>
2021-01-11 15:10:51 +08:00

682 lines
15 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# Api语法描述
## api示例
``` golang
/**
* api语法示例及语法说明
*/
// api语法版本
syntax = "v1"
// import literal
import "foo.api"
// import group
import (
"bar.api"
"foo/bar.api"
)
info(
author: "songmeizi"
date: "2020-01-08"
desc: "api语法示例及语法说明"
)
// type literal
type Foo{
Foo int `json:"foo"`
}
// type group
type(
Bar{
Bar int `json:"bar"`
}
)
// service block
@server(
jwt: Auth
group: foo
)
service foo-api{
@doc "foo"
@handler foo
post /foo (Foo) returns (Bar)
}
```
## api语法结构
* syntax语法声明
* import语法块
* info语法块
* type语法块
* service语法块
* 隐藏通道
> ### 温馨提示️
> 在以上语法结构中,各个语法块从语法上来说,按照语法块为单位,可以在.api文件中任意位置声明
> 但是为了提高阅读效率,我们建议按照以上顺序进行声明,因为在将来可能会通过严格模式来控制语法块的顺序。
### syntax语法声明
syntax是新加入的语法结构该语法的引入可以解决
* 快速针对api版本定位存在问题的语法结构
* 针对版本做语法解析
* 防止api语法大版本升级导致前后不能向前兼容
> ### 警告 ⚠️
> 被import的api必须要和main api的syntax版本一致。
**语法定义**
``` antlrv4
'syntax'={checkVersion(p)}STRING
```
**语法说明**
> syntax固定token标志一个syntax语法结构的开始
>
> checkVersion自定义go方法检测`STRING`是否为一个合法的版本号目前检测逻辑为STRING必须是满足`(?m)"v[1-9][0-9]*"`正则。
>
> STRING一串英文双引号包裹的字符串如"v1"
>
> 一个api语法文件只能有0或者1个syntax语法声明如果没有syntax则默认为v1版本
>
**正确语法示例**
eg1不规范写法
``` api
syntax="v1"
```
eg2规范写法(推荐)
``` api
syntax = "v2"
```
**错误语法示例**
eg1
``` api
syntax = "v0"
```
eg2
``` api
syntax = v1
```
eg3
``` api
syntax = "V1"
```
## import语法块
随着业务规模增大api中定义的结构体和服务越来越多所有的语法描述均为一个api文件这是多么糟糕的一个问题 其会大大增加了阅读难度和维护难度import语法块可以帮助我们解决这个问题通过拆分api文件
不同的api文件按照一定规则声明可以降低阅读难度和维护难度。
> ### 警告 ⚠️
> 这里import不像golang那样包含package声明仅仅是一个文件路径的引入最终解析后会把所有的声明都汇聚到一个spec.Spec中。
> 不能import多个相同路径否则会解析错误。
**语法定义**
``` antlrv4
'import' {checkImportValue(p)}STRING
|'import' '(' ({checkImportValue(p)}STRING)+ ')'
```
**语法说明**
> import固定token标志一个import语法的开始
>
> checkImportValue自定义go方法检测`STRING`是否为一个合法的文件路径目前检测逻辑为STRING必须是满足`(?m)"(/?[a-zA-Z0-9_#-])+\.api"`正则。
>
> STRING一串英文双引号包裹的字符串如"foo.api"
>
**正确语法示例**
eg
``` api
import "foo.api"
import "foo/bar.api"
import(
"bar.api"
"foo/bar/foo.api"
)
```
**错误语法示例** ❌
eg
``` api
import foo.api
import "foo.txt"
import (
bar.api
bar.api
)
```
## info语法块
info语法块是一个包含了多个键值对的语法体其作用相当于一个api服务的描述解析器会将其映射到spec.Spec中 以备用于翻译成其他语言(golang、java等)
时需要携带的meta元素。如果仅仅是对当前api的一个说明而不考虑其翻译 时传递到其他语言则使用简单的多行注释或者java风格的文档注释即可关于注释说明请参考下文的 **隐藏通道**
> ### 警告 ⚠️
> 不能使用重复的key每个api文件只能有0或者1个info语法块
**语法定义**
``` antlrv4
'info' '(' (ID {checkKeyValue(p)}VALUE)+ ')'
```
**语法说明**
> info固定token标志一个info语法块的开始
>
> checkKeyValue自定义go方法检测`VALUE`是否为一个合法值。
>
> VALUEkey对应的值可以为单行的除'\r','\n','/'后的任意字符,多行请以""包裹,不过强烈建议所有都以""包裹
>
**正确语法示例** ✅
eg1不规范写法
``` api
info(
foo: foo value
bar:"bar value"
desc:"long long long long
long long text"
)
```
eg2规范写法(推荐)
``` api
info(
foo: "foo value"
bar: "bar value"
desc: "long long long long long long text"
)
```
**错误语法示例** ❌
eg1没有key-value内容
``` api
info()
```
eg2不包含冒号
``` api
info(
foo value
)
```
eg3key-value没有换行
``` api
info(foo:"value")
```
eg4没有key
``` api
info(
: "value"
)
```
eg5非法的key
``` api
info(
12: "value"
)
```
eg6移除旧版本多行语法
``` api
info(
foo: >
some text
<
)
```
## type语法块
在api服务中我们需要用到一个结构体(类)来作为请求体,响应体的载体,因此我们需要声明一些结构体来完成这件事情, type语法块由golang的type演变而来当然也保留着一些golang type的特性沿用golang特性有
* 保留了golang内置数据类型`bool`,`int`,`int8`,`int16`,`int32`,`int64`,`uint`,`uint8`,`uint16`,`uint32`,`uint64`,`uintptr`
,`float32`,`float64`,`complex64`,`complex128`,`string`,`byte`,`rune`,
* 兼容golang struct风格声明
* 保留golang关键字
> ### 警告 ⚠️
> * 不支持alias
> * 不支持time.Time数据类型
> * 结构体名称、字段名称、不能为golang关键字
**语法定义**
> 由于其和golang相似因此不做详细说明具体语法定义请在[ApiParser.g4](g4/ApiParser.g4)中查看typeSpec定义。
**语法说明**
> 参考golang写法
**正确语法示例** ✅
eg1不规范写法
``` api
type Foo struct{
Id int `path:"id"` // ①
Foo int `json:"foo"`
}
type Bar struct{
// 非导出型字段
bar int `form:"bar"`
}
type(
// 非导出型结构体
fooBar struct{
FooBar int
}
)
```
eg2规范写法推荐
``` api
type Foo{
Id int `path:"id"`
Foo int `json:"foo"`
}
type Bar{
Bar int `form:"bar"`
}
type(
FooBar{
FooBar int
}
)
```
**错误语法示例** ❌
eg
``` api
type Gender int // 不支持
// 非struct token
type Foo structure{
CreateTime time.Time // 不支持time.Time
}
// golang关键字 var
type var{}
type Foo{
// golang关键字 interface
Foo interface
}
type Foo{
foo int
// map key必须要golang内置数据类型
m map[Bar]string
}
```
**① tag说明**
> tag定义和golang中json tag语法一样除了json tag外go-zero还提供了另外一些tag来实现对字段的描述
> 详情见下表。
* tag表
|tag key |描述 |提供方 |有效范围 |示例 |
|:--- |:--- |:--- |:--- |:--- |
|json|json序列化tag|golang|request、response|`json:"fooo"`|
|path|路由path如`/foo/:id`|go-zero|request|`path:"id"`|
|form|标志请求体是一个formPOST方法时或者一个query(GET方法时`/search?name=keyword`)|go-zero|request|`form:"name"`|
* tag修饰符
> 常见参数校验描述
|tag key |描述 |提供方 |有效范围 |示例 |
|:--- |:--- |:--- |:--- |:--- |
|optional|定义当前字段为可选参数|go-zero|request|`json:"name,optional"`|
|options|定义当前字段的枚举值,多个以竖线②隔开|go-zero|request|`json:"gender,options=male"`|
|default|定义当前字段默认值|go-zero|request|`json:"gender,default=male"`|
|range|定义当前字段数值范围|go-zero|request|`json:"age,range=[0:120]"`|
② 竖线:|
> ### 温馨提示
> tag修饰符需要在tag value后以引文逗号,隔开
## service语法块
service语法块用于定义api服务包含服务名称服务metadata中间件声明路由handler等。
> ### 警告 ⚠️
> * main api和被import的api服务名称必须一致不能出现服务名称歧义。
> * handler名称不能重复
> * 路由(请求方法+请求path名称不能重复
> * 请求体必须声明为普通非指针struct响应体做了一些向前兼容处理详请见下文说明
>
**语法定义**
``` antlrv4
serviceSpec: atServer? serviceApi;
atServer: '@server' lp='(' kvLit+ rp=')';
serviceApi: {match(p,"service")}serviceToken=ID serviceName lbrace='{' serviceRoute* rbrace='}';
serviceRoute: atDoc? (atServer|atHandler) route;
atDoc: '@doc' lp='('? ((kvLit+)|STRING) rp=')'?;
atHandler: '@handler' ID;
route: {checkHttpMethod(p)}httpMethod=ID path request=body? returnToken=ID? response=replybody?;
body: lp='(' (ID)? rp=')';
replybody: lp='(' dataType? rp=')';
// kv
kvLit: key=ID {checkKeyValue(p)}value=LINE_VALUE;
serviceName: (ID '-'?)+;
path: (('/' (ID ('-' ID)*))|('/:' (ID ('-' ID)?)))+;
```
**语法说明**
> serviceSpec包含了一个可选语法块`atServer`和`serviceApi`语法块其遵循序列模式编写service必须要按照顺序否则会解析出错
>
> atServer 可选语法块定义key-value结构的server metadata'@server'表示这一个server语法块的开始其可以用于描述serviceApi或者route语法块其用于描述不同语法块时有一些特殊关键key
> 需要值得注意,见 **atServer关键key描述说明**。
>
> serviceApi包含了1到多个`serviceRoute`语法块
>
> serviceRoute按照序列模式包含了`atDoc`,handler和`route`
>
> atDoc可选语法块一个路由的key-value描述其在解析后会传递到spec.Spec结构体如果不关心传递到spec.Spec,
> 推荐用单行注释替代。
>
> handler是对路由的handler层描述可以通过atServer指定`handler` key来指定handler名称
> 也可以直接用atHandler语法块来定义handler名称
>
> atHandler'@handler' 固定token后接一个遵循正则`[_a-zA-Z][a-zA-Z_-]*`)的值用于声明一个handler名称
>
> route路由有`httpMethod`、`path`、可选`request`、可选`response`组成,`httpMethod`是必须是小写。
>
> bodyapi请求体语法定义必须要由()包裹的可选的ID值
>
> replyBodyapi响应体语法定义必须由()包裹的struct、~~array(向前兼容处理后续可能会废弃强烈推荐以struct包裹不要直接用array作为响应体)~~
>
> kvLit 同info key-value
>
> serviceName: 可以有多个'-'join的ID值
>
> pathapi请求路径必须以'/'或者'/:'开头,切不能以'/'结尾中间可包含ID或者多个以'-'join的ID字符串
**atServer关键key描述说明**
修饰service时
|key|描述|示例|
|:---|:---|:---|
|jwt|声明当前service下所有路由需要jwt鉴权且会自动生成包含jwt逻辑的代码|`jwt: Auth`|
|group|声明当前service或者路由文件分组|`group: login`|
|middleware|声明当前service需要开启中间件|`middleware: AuthMiddleware`|
修饰route时
|key|描述|示例|
|:---|:---|:---|
|handler|声明一个handler|-|
**正确语法示例** ✅
eg1不规范写法
``` api
@server(
jwt: Auth
group: foo
middleware: AuthMiddleware
)
service foo-api{
@doc(
summary: foo
)
@server(
handler: foo
)
// 非导出型body
post /foo/:id (foo) returns (bar)
@doc "bar"
@handler bar
post /bar returns ([]int)// 不推荐数组作为响应体
@handler fooBar
post /foo/bar (Foo) returns // 可以省略'returns'
}
```
eg2规范写法推荐
``` api
@server(
jwt: Auth
group: foo
middleware: AuthMiddleware
)
service foo-api{
@doc "foo"
@handler: foo
post /foo/:id (Foo) returns (Bar)
}
service foo-api{
@handler ping
get /ping
@doc "foo"
@handler: bar
post /bar/:id (Foo)
}
```
**错误语法示例** ❌
``` api
// 不支持空的server语法块
@server(
)
// 不支持空的service语法块
service foo-api{
}
service foo-api{
@doc kkkk // 简版doc必须用英文双引号引起来
@handler foo
post /foo
@handler foo // 重复的handler
post /bar
@handler fooBar
post /bar // 重复的路由
// @handler和@doc顺序错误
@handler someHandler
@doc "some doc"
post /some/path
// handler缺失
post /some/path/:id
@handler reqTest
post /foo/req (*Foo) // 不支持除普通结构体外的其他数据类型作为请求体
@handler replyTest
post /foo/reply returns (*Foo) // 不支持除普通结构体、数组(向前兼容,后续考虑废弃)外的其他数据类型作为响应体
}
```
## 隐藏通道
隐藏通道目前主要为空百符号,换行符号以及注释,这里我们只说注释,因为空白符号和换行符号我们目前拿来也无用。
### 单行注释
**语法定义**
``` antlrv4
'//' ~[\r\n]*
```
**语法说明**
由语法定义可知道,单行注释必须要以`//`开头,内容为不能包含换行符
**正确语法示例** ✅
``` api
// doc
// comment
```
**错误语法示例**
``` api
// break
line comments
```
### java风格文档注释
**语法定义**
``` antlrv4
'/*' .*? '*/'
```
**语法说明**
由语法定义可知道,单行注释必须要以`/*`开头,`*/`结尾的任意字符。
**正确语法示例**
``` api
/**
* java-style doc
*/
```
**错误语法示例** ❌
``` api
/*
* java-style doc */
*/
```
## Doc&Comment
如果想获取某一个元素的doc或者comment开发人员需要怎么定义
**Doc**
> 我们规定上一个语法块非隐藏通道内容的行数line+1到当前语法块第一个元素前的所有注释(当行,或者多行)均为doc 且保留了`//`、`/*`、`*/`原始标记。
**Comment**
> 我们规定当前语法块最后一个元素所在行开始的一个注释块(当行,或者多行)为comment 且保留了`//`、`/*`、`*/`原始标记。
语法块Doc和Comment的支持情况
|语法块|parent语法块|Doc|Comment|
|:---|:---|:---|:---|
|syntaxLit|api|✅|✅|
|kvLit|infoSpec|✅|✅|
|importLit|importSpec|✅|✅|
|typeLit|api|✅|❌|
|typeLit|typeBlock|✅|❌|
|field|typeLit|✅|✅|
|key-value|atServer|✅|✅|
|atHandler|serviceRoute|✅|✅|
|route|serviceRoute|✅|✅|
以下为对应语法块解析后细带doc和comment的写法
``` api
// syntaxLit doc
syntax = "v1" // syntaxLit commnet
info(
// kvLit doc
author: songmeizi // kvLit comment
)
// typeLit doc
type Foo {}
type(
// typeLit doc
Bar{}
FooBar{
// filed doc
Name int // filed comment
}
)
@server(
/**
* kvLit doc
* 开启jwt鉴权
*/
jwt: Auth /**kvLit comment*/
)
service foo-api{
// atHandler doc
@handler foo //atHandler comment
/*
* route doc
* post请求
* path为 /foo
* 请求体Foo
* 响应体Foo
*/
post /foo (Foo) returns (Foo) // route comment
}
```