Skip to content

Pagination the REST way : 'Content-Range' header #7137

Open
@Nayte91

Description

@Nayte91

Hello, I move this post from discussions (#2804) to RFC issue, because I'll try to implement it,

I was lurking on the internet when I wondering how to implement easily and in a generic way a count for a given resource, the REST way?

For example if you have a dictionary API and you already have a 'count' word definition, it's unconvenient to add a api.foo/words/count route. It's also not very RESTful to create and impose the usage of a new verb COUNT, dedicated for this operation.

What we want, ideally, is to have the number of items in the pagination of this current request, ('30 items' in your book demo API), how many resources can be found on every page ('100 items' in total?), and therefor you can calculate how much pages there is. Also, for economic purpose, we want the content of the response to be empty or totally different of the other call on this resource, for example with just those counts.

And I feel like there is a very nice way to deal with this: as a user, you can request your book/ resource with the HEAD verb, that will ensure no body in response, and as an API provider, you can enrich your response with this nice 'Content-Range' header!

HEAD verb

HEAD works like... HEAD, it's already handled, so nothing to say more.

Content-Range HTTP header

'Content-Range' header is intended to carry a bunch of very clever information, as the common example is:

Content-Range: bytes 200-1000/67589

Where we can find the unit, the current range, and / the total count. Implicitely, you can find the number of items served in this response, here 1000-200 = 800 bytes, the current page, and the total number of pages. The tricky part is that the <unit> is usually "bytes", but the specifications doesn't limit!

So there's a scenario where API-Platform can answer a Content-Range: books 1-30/201 when requesting 'https://demo.api-platform.com/books':

  • The <unit> is the resource on plural, an information API-Platform already has when naming routes,
  • The <range-start>-<range-end> is defined by pagination,
  • The <size> is the total count of this request.

I feel like there is some MASSIVE advantages in it:

  • very convenient info about current pagination, even in regular JSON answers (as hydra returns a hydra:totalItems key for it),
  • clever usage of HEAD verb for economic requests as you don't need to charge all the jsonLD payload to have info,
  • native HTTP solution that totally adheres in REST,
  • saves few headaches of 'how will provide a count for this request on my API?' with an out of the box solution; JSON+LD advantages for everyone!

You can argue that if specifications don't limit unit to be bytes, they don't limit verbs also; The point is that adding a custom verb like COUNT force requesters to know it, firewalls to accept it. On the other hand, Content-Range header usage with a 'resource' as unit is a flat win for requester, doesn't break current API and won't break anything in any system dealing with the answer.

I feel like it's enough for a first implementation (I may be very wrong, but as json+ld implementation already has those info, it's more like populating this header), but in future a nice improvement is Range: header can also be used instead of a page query parameter, to have a complete pagination handling in headers. It allows also to give some love to the Accept-Ranges: header and maybe the OPTION verb to document how to deal with pagination (ranges is the correct name in fact)!

Resources (no pun intended):

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions