在上一篇中:RESTful API 设计指南——开篇词
我们介绍了几个十分有争议的案例:
本章我们来深入分析一下,为什么不要像案例中所说的那样干?
RESTful 规范能解决这些问题吗?
我们首先来看一下 GET 和 POST 的区别(来源文章):
通过上面的对比,我们看到 GET 和 POST 都有各自的使用场景,假设你要做一个分享功能,或者一个博客,那么可以被添加为书签或者能直接在浏览器地址粘贴打开是必须的,否则将会丢失很多从收藏夹过来的阅读量。
所以,在早期的 MVC 时代(前后端在一起),肯定不能所有的请求都是 POST,否则有一些功能将会无法实现。
那么在当下前后端分离的时代呢?后端接口通常是提供给前端使用的,用户不会直接接触到这些后端接口,可以都使用 POST 请求吗?
在前后端分离架构下,团队分工发生了变化:由 Java 干所有事情,变成了前端开发 + 后端开发的组合,这意味分工的精细化,前端除了功能实现,还会关注界面美观和用户体验;后端会关注性能、并发和可用性等一些非功能需求。从这个角度而言 GET 能使用 CDN 缓存,限制 GET 只用来查询,POST用来修改,从而可以通过读写分离机制提升整体服务的性能和可用性就很关键了。
所以,在前后端分离和微服务架构的时代的今天,功能只是整个后端开发中的一部分,有时甚至非功能需求的工作量远远大于功能性需求,所以,不太可能出现所有接口都使用 POST 请求这种情况。
另外,伴随着云计算时代的到来, K8S 和 Docker 容器化部署几乎成为了后端的必备技能之一,而 K8S 其中的一个机制:健康检查就要求服务提供一个 HTTP GET 接口,返回200 OK 代表服务已准备就绪,K8S 才会开放流量给该进程。所以,也不可能只提供 POST 接口的。
当然,微服务中还会伴随各种疑难问题,比如监控、压测、流控、路由和权限等等,可以参考这篇文章了解:来源文章,也没办法只使用 POST 请求。
在上一篇文章: RESTful API 设计指南——开篇词 中,我们有说过,如果不区分 HTTP 状态码,不管成功还是失败都返回200 OK,这样做最大的问题是:监控系统在一种低效的状态下工作。
原因是监控系统需要解析你的 Body,才能知道是调用方出错还是服务端。这里的监控系统只是一个概念,泛指指基础设施或者第三方监控系统,可以是 nginx、k8s 的 ingress 、api 网关(kong 和 Spring Cloud Gateway等)或者 promethus 等等。
这里我们以大家熟知的 Nginx 举例来说明,它在什么地方会用到 HTTP 状态码。
Nginx是主流的Web服务器之一,除了可以做 HTTP 或 TCP 代理之外,最常用的一个功能就是做负载均衡。
Nginx 中实现负载均衡的是 upstream(ngx_http_upstream_module) 模块,比如下面的例子:
upstream test {
server 10.78.3.1:80 fail\_timeout=60s max\_fails=2; # Server A
server 10.78.3.2:80 fail\_timeout=60s max\_fails=2; # Server B
server 10.78.3.3:80 backup; # Server C
}
upstream 是一个非常强大的模块,除了轮询负载算法还支持 ip_hash、fair、url_hash 等算法,另外每个节点支持很多配置:
更多关于 upstream模块 的介绍可以参考官网和这篇文章:nginx upstream配置说明负载均衡
upstream 模块通常搭配 http代理模块(ngx_http_proxy_module)使用,用来实现 HTTP 代理转发(还支持4层代理,如TCP和UDP等),比如下面的例子:
http {
# 负载均衡,名称是 test
upstream test {
server 10.78.3.1:80 fail\_timeout=60s max\_fails=2; # Server A
server 10.78.3.2:80 fail\_timeout=60s max\_fails=2; # Server B
server 10.78.3.3:80 backup; # Server C
}
server {
listen 9998;
server\_name pub\_api;
# 转发到 test 负载均衡上
location / {
root html;
proxy\_pass ://test>;
}
}
}
我们配置了一个具有3个节点的 Web 服务器集群,通过 9998 端口统一对外暴露服务,这样上层系统就只需要知道一个 IP 地址,并且当其中某一个节点出问题时,Nginx 会通过负载均衡算法帮我们把请求转发到其他可用的节点上。
那这里和 HTTP 状态码有什么关系呢?
在上面的 location 节点中,我们还可以配置各种超时和重试策略:
location / {
root html;
proxy\_connect\_timeout 5s;
proxy\_read\_timeout 100s;
proxy\_send\_timeout 70s;
proxy\_next\_upstream error timeout;
proxy\_next\_upstream\_timeout 70;
proxy\_next\_upstream\_tries 1;
proxy\_pass ://efast-public>;
}
我们着重看一下失败重试的配置项 proxy_next_upstream。
根据 官方的文档,它的语法如下:
Syntax:proxy\_next\_upstream error | timeout | invalid\_header | http\_500 | http\_502 | http\_503 | http\_504 | http\_403 | http\_404 | http\_429 | non\_idempotent | off ...;
Default:proxy\_next\_upstream error timeout;
Context:http, server, location
发现了吗?
它可以根据 HTTP 状态码来决定是否需要重试,如果我们不区分 HTTP 状态码,不管成功还是失败,是客户端参数导致的错误,还是服务器内部的错误都返回200的话,我们是没有办法使用 nginx 失败重试的功能的,除非你修改它这个模块的源码!
还记得上文提到过的API命名问题吗?
以新增员工为例,因为开发习惯不同,可能会出现这些情况:
http://localhost/employee/save
http://localhost/employee/add
http://localhost/employee/new
http://localhost/employee/xinzeng
http://localhost/employee/append
http://localhost/employee?cmd=add
http://localhost/add_employee
除了上述问题之外,这个新增员工接口是发 POST 还是 GET 请求?是通过 HTTP 状态码代表操作结果,还是需要解析 json 来判断?如果是解析 json ,那个字段代表成功?等等这些问题,需要我们仔细的查阅它的 API 文档才能下定论。
那么,RESTful API 是如何解决这些的问题的呢?
在《RESTful Web Services》这本书的第4章:面向资源的架构中,为我们做出了解答:统一接口!
客户端与资源之间的所有交互,都是通过为数不多的几个HTTP方法进行的。任何资源都将暴露这些方法的一个子集,而且这些方法无论被哪个资源暴露,都具有相同的意义。
即对于任何资源,增删改查统一使用 HTTP 方法来表示:
如下是一个使用 HTTP 方法来代表操作的一个示例:
请求方法 | URL | 描述 |
---|---|---|
GET | /departments | 获取所有部门信息 |
POST | /departments | 新增一个新的部门 |
GET | /departments/{departmentid} | 获取指定部门详细信息 |
PUT | /departments/{departmentid} | 更新指定部门信息 |
DELETE | /departments/{departmentid} | 删除指定部门 |
所以,按照 RESTful API 规范,新增操作必须发送 POST 请求,至于具体是新增什么,需要由 url 来决定:
POST /employees # 新增员工
POST /departments # 新增部门
POST /roles # 新增角色
至此,增删改查接口命名的问题解决了,在 RESTful API 中,看到 POST 就代表这是一个新增接口,看到 GET 就知道这个接口是用来查询的。至于具体的含义,需要结合资源名(path)来看。另外,资源通常使用名词表示,也就意味着 path 中不应该出现动词,关于这些我们会在后续的文章中详细说明。
本章我们详细介绍了在 HTTP API 设计中可能会遇到的一些典型的非常具有代表性和争议的案例,我们应该极力避免案例中的做法。
对于“所有的接口都使用Post请求”的案例,我们给出了足够多避免这样做的理由,比如 POST 和 GET 的适用的场景不同,健康检查需要提供 GET 请求、不利于读写分离和监控系统配合等。
对于"不管成功还是失败,HTTP状态码都返回200”的案例,我们也通过介绍 Nginx 的 HTTP 代理功能,介绍了微服务时代下,服务和基础设施之间配合的问题。
对于“API命名千奇百怪”的案例,我们介绍了为什么引入 REST 规范能避免该问题,因为 RESTful API 统一了操作接口,增删改查使用 HTTP 方法表示,从此,对于各个团队,大家的玩法都一样了。
在本章,我们介绍了使用 RESTful API 风格的带来的收益和好处,下一章中,我们会介绍一下它的劣势,毕竟,软件世界中没有银弹。
作者简介:一线Coder,公众号《Go和分布式IM》运营者,开源项目: CoffeeChat 、interview-golang 发起人 & 核心开发者,终身程序员。