本规范的三个目标:简洁、统一、开放。
关于如何设计良好风格的 RESTful API,Github 有一份满分答案,熟读三遍,其义自现。本规范将在其基础之上使用尽可能简单的表述方式从以下几个常见部分作出详细约定:
注意本规范是笔者在所在公司内部制定的规范整理而来的接口规范模板,仅供你在制定规范时候参考,在落地时应根据团队实际情况作出调整,欢迎对本规范进行补充
/api
或 /[version]/api
开头正确:/api/task
或 /v2/api/tasks
错误:/biz/tasks
或 /biz/api/tasks
注意:一个产品无论后端有多少个服务组成也应该只有一个 API 入口
api/aa-bb/cc-dd
方式命名正确:/api/task-groups
错误:/api/taskGroups
正确:POST /api/tasks
或 /api/task-groups/1/tasks
表示在 id 为 1 的任务组下创建任务
错误:POST /api/create-task
正确:/api/tasks
错误:/api/task
要求我们在给结构路径命名时候面向通用业务的开放接口,而非单纯前端业务,以获取筛选表单中的任务字段下拉选项为例
正确:/api/tasks
错误:/api/task-select-options
虽然这个接口暂时只用在表单的下拉选择中,但是需要考虑的是在未来可能会被用在任意场景,因此应以更通用方式提供接口交由客户端自由组合
方法 | 场景 | 例如 |
---|---|---|
GET | 获取数据 | 获取单个:GET /api/tasks/1、获取列表:GET /api/tasks |
POST | 创建数据 | 创建单个:POST /api/tasks |
PATCH | 差量修改数据 | 修改单个:PATCH /api/tasks/1 |
PUT | 全量修改数据 | 修改单个:PUT /api/tasks/1 |
DELETE | 删除数据 | 删除单个:DELETE /api/tasks/1 |
其它更多请求方法请查阅 MDN Web Docs
状态码 | 场景 |
---|---|
200 | 创建成功,通常用在同步操作时 |
202 | 创建成功,通常用在异步操作时,表示请求已接受,但是还没有处理完成 |
400 | 参数错误,通常用在表单参数错误 |
401 | 授权错误,通常用在 Token 缺失或失效,注意 401 会触发前端跳转到登录页 |
403 | 操作被拒绝,通常发生在权限不足时,注意此时务必带上详细错误信息 |
404 | 没有找到对象,通常发生在使用错误的 id 查询详情 |
500 | 服务器错误 |
其它更多响应状态码请查阅 MDN Web Docs
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '任务 1'
}
}
{
code: 20000,
status: 200,
message: "请求成功",
data: {
items: [{
id: 1,
name: '任务 1'
}, {
id: 2,
name: '任务 2'
}],
total: 2
}
}
注意:其中 code
表示业务编码,status
表示 HTTP 响应状态码,如此设计的原因是部分场景下前后端之间存在不可控的网关或代理(比如某些网关有一些流量控制策略会导致直接返回 403 响应状态码,此时客户端无法分辨 403 是网关的还是业务方),类似这类情况下为了能够让客户端正确分辨业务方的真实处理结果则需要在响应体加上 status
,而 code
表示的业务编码是为了帮助工程师更容易定位问题,它并不是必须的,这取决你们团队风格。
尽管在响应体中体现了响应状态码,但这并不代表所有 HTTP 就可以全部返回 200 了,无论如何请在条件允许范围内尽可能使用标准的 HTTP 响应状态码
aa_bb_cc
方式命名// 正确
{
role_ids: [11,12,35],
}
// 错误
{
roleIds: [11, 12, 35],
RoleIds: [11, 12, 35],
ROLE_IDS: [11, 12, 35]
}
YYYY-MM-DDTHH:MM:SSZ
名称:name
状态:status
创建时间:created_at
更新时间:updated_at
// 正确
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
role_ids: [],
}
}
// 错误
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
role_ids: null
}
}
// 正确
{
code: 20000,
status: 200,
message: "请求成功",
data: {
roles: [{
id: 1,
name: '角色 1'
}, {
id: 2,
name: '角色 2'
}]
}
}
// 错误
{
code: 20000,
status: 200,
message: "请求成功",
data: {
roles: '[{"id":1,"name":"角色 1"},{"id":2,"name":"角色 2"}]'
}
}
{
code: 20000,
status: 200,
message: "创建成功",
data: {
id: 1,
}
}
以创建用户为例:POST /api/users
// 正确
{
username: 'ming'
password: 'xxxx',
role_ids: [1, 2, 3]
}
// 错误
{
username: 'ming'
password: 'xxxx',
role_ids: [{
id: 1,
name: '角色1'
}, {
id: 2,
name: '角色2'
}, {
id: 3,
name: '角色3'
}]
}
{
code: 40000
status: 400;
message: "参数错误",
data: {
errors: [{
field: 'name',
message: '缺失'
}]
}
}
例如 GET /api/tasks?sort=created_at&order=descend
表示以创建时间降序查询数据
注意:其中 order
为 descend
时表示降序,为 ascend
时表示升序
例如 GET /api/tasks?page=1&per_page=10
表示每页 10 条查询第一页数据
注意:其中 page
从 1 开始,而不是 0,如果没有传递 per_page
和 page
参数表示不分页获取所有数据
keyword
关键词,枚举筛选使用数组合并拼接,区间使用 xxx_lt
和 xxx_gt
关键词例如 GET /api/tasks?creator=ming
表示查询所有 ming 用户创建的任务
例如 GET /api/tasks?keyword=ming
表示查询任意列包含 ming 关键词的任务
例如 GET /api/tasks?status=pending,complete
表示查询状态为阻塞和完成的任务
例如 GET /api/tasks?weight_gt=10&weight_lt=20
表示查询权重在 10 和 20 之间的任务
例如 GET /api/tasks?weight_gt=10
表示查询权重大于 10 的任务
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
username: 'ming'
roles: [{
id: 1,
name: '角色 1'
}, {
id: 2,
name: '角色 2'
}, {
id: 3,
name: '角色 3'
}]
}
}
// 正确
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '任务 1'
status: 'pending' // 'pending' | 'complete' | 'error'
}
}
// 错误
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '任务 1'
status: 0
}
}
// 正确
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '任务 1'
creator: {
id: 1,
username: '小明'
}
}
}
// 错误
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '任务 1'
creator_id: 1,
creator_name: '小明'
}
}
例如 DELETE /api/tasks/1
表示删除 id 为 1 的任务
例如 DELETE /api/tasks?ids=1,2,3
表示批量删除 id 为 1 或 2 或 3 的任务
注意:如果列表数据量较大或容易沉淀无用数据的应提供批量删除功能,比如任务、文件、日志等
/api/files
),支持上传所有类型文件// 请求,注意这里是 FormData
{
file: File
}
// 响应
{
code: 20000,
status: 200,
message: "上传成功",
data: {
id: 'bb313c99',
url: '/files/bb313c99.pdf'
name: '合同.pdf' // 原文件的名称
}
}
/api/multiple-files
),支持上传所有类型文件// 请求,注意这里是 FormData
{
files: [File, File]
}
// 响应
{
code: 20000,
status: 200,
message: "上传成功",
data: [{
id: 'bb313c99',
url: '/files/bb313c99.pdf'
name: '合同1.pdf' // 原文件的名称
}, {
id: 'bb313c88',
url: '/files/bb313c88.pdf'
name: '合同2.pdf' // 原文件的名称
}]
}
// 正确
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '小明'
avatar: '/files/bb313c99.png',
// 或
avatar: 'https://cdn.xxx.com/files/bb313c99.png',
}
}
// 错误
{
code: 20000,
status: 200,
message: "请求成功",
data: {
id: 1,
name: '小明'
avatar: 'bb313c99.png'
}
}
// 正确
{
name: '新任务 1',
file_id: 'bb313c99',
// 或
file_url: '/files/bb313c99.pdf',
}
// 错误
{
name: '新任务 1',
file: File
}
注意:先由 POST api/files
上传完文件拿到文件 id 或地址后再执行后续操作
// 正确
{
name: '小明',
id_number: 'U2FsdGVkX1+1fW7OpO/tlPXe4IGA/bXExlhKwIR/spk=',
password: 'U2FsdGVkX1/AnXKSBDbztNBfp4czlZxQ++3jRtNZhY0=',
}
// 错误
{
name: '小明',
id_number: '310000199511159999',
password: '[email protected]',
}
注意:本规范不约定使用何种加密算法,请视实际场景选择
{
code: 20000,
status: 200,
message: "请求成功",
data: {
x_axis: ['2022.04.20','2022.04.21', '2022.04.22']
series: [{
name: '上海用户',
data: [5000,4000,3000],
color: '#f5f5f5' // 可选,如果加上的话会使用该色值
}, {
name: '成都用户',
data: [3000,4000,5000], // 注意,没有数据时候也要使用 0 填充,和 x_axis 一一对应
color: '#f5f5f5'
}]
}
}
{
code: 20000,
status: 200,
message: "请求成功",
data: {
series: [{
name: '上线用户',
value: 1890,
color: '#f5f5f5' // 可选,如果加上的话会使用该色值
}, {
name: '下线用户',
value: 2000,
color: '#f5f5f5'
}]
}
}