--- title: RestFul接口规范 date: 2020-06-24 16:30:08 tags: [接口测试] categories: [接口测试] author: 阮一峰 --- ## 什么是RestFul? RestFul是在2000年由Roy Thomas Fielding博士在他的博士论文中提出。 他是http 1.0和1.1版本协议的主要设计者,apache基金会的第一任主席,可以说是现代互联网体系的奠基者。 Fielding将他对互联网软件的架构原则,定名为REST,即表述层状态转移(Representational State Transfer)。 这是一套在互联网体系下,调用者如何与被调用者(资源实体)进行互动的规范设计。 ## RestFul有什么样的规范? 在RestFul理论中,一个uri地址,指向一个资源对象。 ### 设计 以不同的请求方式,来实现资源对象的不同操作。 GET :从服务器取出资源(一项或多项) POST :在服务器新建一个资源 PUT :在服务器更新资源(客户端提供改变后的完整资源) PATCH :在服务器更新资源(客户端提供改变的属性) DELETE :从服务器删除资源 例如: 有一个资源对象为user,1 是资源对象中的某一个对象中的唯一标识id。 ``` GET /user:返回所有用户对象的列表(数组) GET /user/resource:返回单个用户对象 POST /user:返回新生成的用户对象 PUT /user/resource:返回完整的用户对象 PATCH /user/resource:返回完整的用户对象 DELETE /user/resource:返回一个空文档 ``` ### 复数url 既然 URL 是名词,那么应该使用复数,还是单数? 这没有统一的规定,但是常见的操作是读取一个集合,比如GET /articles(读取所有文章),这里明显应该是复数。 为了统一起见,建议都使用复数 URL,比如GET /articles/2要好于GET /article/2。 ### 避免多级URL 常见的情况是,资源需要多级分类,因此很容易写出多级的 URL,比如获取某个作者的某一类文章。 ``` GET /authors/12/categories/2 ``` 这种 URL 不利于扩展,语义也不明确,往往要想一会,才能明白含义。 更好的做法是,除了第一级,其他级别都用查询字符串表达。 ``` GET /authors/12?categories=2 ``` 下面是另一个例子,查询已发布的文章。你可能会设计成下面的 URL。 ``` GET /articles/published ``` 查询字符串的写法明显更好。 ``` GET /articles?published=true ``` ### 状态码 状态码必须精确 客户端的每一次请求,服务器都必须给出回应。回应包括 HTTP 状态码和数据两部分。 HTTP 状态码就是一个三位数,分成五个类别。 1xx:相关信息 2xx:操作成功 3xx:重定向 4xx:客户端错误 5xx:服务器错误 这五大类总共包含100多种状态码,覆盖了绝大部分可能遇到的情况。 每一种状态码都有标准的(或者约定的)解释,客户端只需查看状态码,就可以判断出发生了什么情况, 所以服务器应该返回尽可能精确的状态码。 API 不需要1xx状态码。 状态码详情可以看[常见HTTP状态码](http://www.angeszhu.cn/auto/api/http-status.html) ### 服务器响应 #### 不要返回纯本文 API 返回的数据格式,不应该是纯文本,而应该是一个 JSON 对象,因为这样才能返回标准的结构化数据。 所以,服务器回应的 HTTP 头的Content-Type属性要设为application/json。 客户端请求时,也要明确告诉服务器,可以接受 JSON 格式,即请求的 HTTP 头的ACCEPT属性也要设成application/json。下面是一个例子。 ``` GET /orders/2 HTTP/1.1 Accept: application/json ``` #### 发生错误时,不要返回 200 状态码 有一种不恰当的做法是,即使发生错误,也返回200状态码,把错误信息放在数据体里面,就像下面这样。 ``` HTTP/1.1 200 OK Content-Type: application/json { "status": "failure", "data": { "error": "Expected at least two items in list." } } ``` 上面代码中,解析数据体以后,才能得知操作失败。 这张做法实际上取消了状态码,这是完全不可取的。 正确的做法是,状态码反映发生的错误,具体的错误信息放在数据体里面返回。下面是一个例子。 ``` HTTP/1.1 400 Bad Request Content-Type: application/json { "error": "Invalid payoad.", "detail": { "surname": "This field is required." } } ``` #### 提供链接 API 的使用者未必知道,URL 是怎么设计的。 一个解决方法就是,在回应中,给出相关链接,便于下一步操作。 这样的话,用户只要记住一个 URL,就可以发现其他的 URL。这种方法叫做 HATEOAS。 举例来说,GitHub 的 API 都在 api.github.com 这个域名。访问它,就可以得到其他 URL。 ``` { ... "feeds_url": "https://api.github.com/feeds", "followers_url": "https://api.github.com/user/followers", "following_url": "https://api.github.com/user/following{/target}", "gists_url": "https://api.github.com/gists{/gist_id}", "hub_url": "https://api.github.com/hub", ... } ``` 上面的回应中,挑一个 URL 访问,又可以得到别的 URL。 对于用户来说,不需要记住 URL 设计,只要从 api.github.com 一步步查找就可以了。 HATEOAS 的格式没有统一规定,上面例子中,GitHub 将它们与其他属性放在一起。 更好的做法应该是,将相关链接与其他属性分开。 ``` HTTP/1.1 200 OK Content-Type: application/json { "status": "In progress", "links": {[ { "rel":"cancel", "method": "delete", "href":"/api/status/12345" } , { "rel":"edit", "method": "put", "href":"/api/status/12345" } ]} } ``` 原文链接:[RESTful API 最佳实践](http://www.ruanyifeng.com/blog/2018/10/restful-api-best-practices.html)