在我的REST API中,我有一个很大的集合;它包含数百万个项目。该集合的路径为/mycollection
由于此集合太大,因此GET
整个集合不是一个好习惯,因此API支持分页。分页将是获取收藏的主要方式
GET /mycollection?page=1&page-size=100 HTTP/1.1
说原始集合包含1,000,000个项目,而我想更新5,000,删除3,000和添加2,000个项目。我可以编写自己的API以支持通过PUT
方法或PATCH
方法更新集合。尽管这两种方法都需要非常不同的请求主体,但我相信这两种方法都需要完全相同的响应主体,即,响应主体必须包含整个更新资源的当前表示形式,即集合中的所有999,000项。
正如我之前提到的GET
仅对整个集合进行评估是不现实的;这个太大了。由于相同的原因,我不希望PUT
或PATCH
归还整个收藏。将查询参数添加到PUT
或PATCH
请求中将不会起作用,因为PUT
和PATCH
都不是安全的方法。
那么在这种大收集情况下,正确的响应是什么?
我可以回应
HTTP/1.1 202 Accepted
Location: /mycollection?page=1&page-size=100
202 Accepted
响应代码不正确,因为更新将同步完成。 Location
标头也不太正确。我也许可以使用Links
标头,但感觉仍然不合适。
再次,我问在这种大收集情况下正确的反应是什么?
答案 0 :(得分:1)
这个问题是基于一个误解:
虽然两种方法都需要非常不同的请求主体,但我相信两种方法都需要完全相同的响应主体,即,响应主体必须包含整个更新资源的当前表示形式
任何人都只能返回204 No Content
或200 OK
而没有响应正文。不需要在响应中包含完整的表示形式。
您可以选择支持此功能(可能与Prefer: return=representation
标头或Content-Location
标头一起使用),但是如果没有此标头,我会说返回当前表示形式甚至不是约定。通用客户端不应假定响应主体是新的表示形式,除非使用了这些标头。
所以,只需返回一个2xx
,就可以了。
答案 1 :(得分:0)
那么在这种大收集情况下,正确的响应是什么?
简短版本:您应该将成功的PUT视为成功的POST。
有效载荷的预期含义可以概括为: 动作状态或结果的代表
所以响应可能很简单
200 OK
Content-Type: text/plain
It worked!
更长的答案:
虽然两种方法都需要非常不同的请求主体,但我相信两种方法都需要完全相同的响应主体,即,响应主体必须包含整个更新资源的当前表示形式
这是不对的-如果您查看RFC 7231,将会看到对PUT的回复中包含以下说明
动作状态的表示
返回资源的新表示形式是一种极端情况,而不是默认情况(请参见Content-Location标头的规范)。
对于诸如PUT(第4.3.4节)或POST(第4.3.3节)之类的状态更改请求,这意味着服务器的响应包含该资源的新表示形式,从而将其与可能仅报告有关以下内容的表示形式区分开来:动作(例如“成功了!”)。这样一来,创作应用程序即可更新其本地副本,而无需后续的GET请求。
也就是说,我建议您回顾一下方法标记的选择。 PUT和PATCH均支持远程创作语义-要求服务器使其文档副本像本地副本一样的消息。这就是例如PUT规范在将验证器标头字段添加到响应中有很多约束的原因。通用组件可以假定它们知道发生了什么,因为所有 资源应该以相同的方式理解这些方法。
但是在您的情况下,您不能真的说您是远程创作该集合,因为客户端(和通用组件)没有该集合的表示,而是只有该页面的表示集合。
如果您要与统一界面保持一致,则可以选择
It is okay to use POST,当您的请求的语义与标准含义不一致时
POST在HTTP中有许多有用的用途,包括“此操作不值得标准化”的一般用途。