- Examples
- Terminology
- Representations
- Authentication
- Status Codes
- Resources
- Collections
- Errors
- Links
- Schemas
- Actions
- Operations
- Nesting
- Resource Versioning
- Base URL and Versioning
- Design Considerations
Examples demonstrate a hypothetical file storage API. Only the relevant HTTP request and response information is shown, additional standard HTTP headers will be present in a real request. Examples generally show JSON request and responses, but several request types are available and other response types may be defined in the future.
REST: A REpresentational State Transfer service is a style of API where a client makes HTTP requests to manipulate resources identified by the request URL. RESTful services are stateless, so no session state is stored on the server between requests. Each requests contains all the information needed to service that request.
Resource: A resource is an object or concept that can be manipulated by the API. In our file storage example, they will be things like a 'Folder' and a 'File'. Resources are the building blocks of an API, the nouns in the language describing the system. All operations in the API manipulate the resources or their state. Resources should be organized in a way that is useful to the client. This is the most important part of the design of an API; if your resources just match your database schema one-to-one, chances are your API is not going to be very easy to use.
Representation: REST APIs transfer representations of resources back and forth between the client and the service. These are documents that describe resources. Representations are most commonly JSON-formatted, but can be of any type.
Attribute: Representations of resources consist of a series of attributes names and values as key-value pairs.
Collection: A collection is a special kind of resource that represents a group of related resources. It contains a representation of each resource, and some additional information for things like pagination and filtering.
Identifier: Every resource in a collection MUST have an identifier attribute with a value that is unique within that collection.
Side Effects: Requests that change the state of something in the service are said to have side effects. It is very important that certain types of requests not have side-effects.
Idempotent: An operation is idempotent if it can be applied multiple times without changing the result after the first application.
Opaque: Some pieces of information returned from a service are considered to be opaque. Opaque values are a black box that has meaning only to the service; clients should not rely on any particular format, length, or content in an opaque value. They should not change the value or try to parse any meaning out of it. Opaque values MAY change between API revisions or even individual requests, so they should never be stored or compared against. For example, pagination URLs contain a "marker" attribute that identifies the appropriate page to load.
Action: An action is an operation that can be performed on a resource but does not fit into one the standard create/read/update/delete operations. For example "sharing" or "encrypting" a file resource.
Discoverability: Discoverability is a method of self-documenting. Each API has a collection of schemas that define what resource types are available and what attributes they have. Each response contains information about how to find related resources and what actions are available.
A representation is the way a resource is described/serialized in a HTTP request or response. All services MUST support:
- JSON (application/json) for both requests and responses.
- This is the primary representation used by all APIs.
- HTML (text/html; charset=utf8) for responses
- This is a small HTML wrapper around the JSON format which presents the HTML UI.
- The HTML response format is only intended for use in a web browser, not for clients to parse.
- Forms (multipart/form-data and application/x-www-form-urlencoded) for requests.
- These are needed for the HTML UI and are easy to create with browsers and command-line tools like cURL.
Services MAY support other representations that make sense for the particular application. These will be defined in the service documentation which a client can specify by sending the value in the Accept
header of the request.
The representation is returned in the Content-Type
header of the response.
Dates MUST be represented as an ISO-8601 formatted string.
- If a time component is included, a time zone designator MUST BE included and SHOULD always be UTC ("Z").
- If you have a very good reason to use something other than UTC, the attribute name SHOULD indicate that it is "local", "user", or similar to avoid confusion.
- e.g. "birthday": "1982-02-24", "created": "2012-09-27T18:39:53Z", "yourLocalTime": "2013-09-27T11:30:42-07:00"
Clients MAY request a preferred representation by including an appropriate Accept header.
- Service SHOULD be lenient in mapping this to one of their acceptable formats.
- For example
application/json
,text/json;charset=utf-8
, andtext/json
can all be interpreted as JSON.
If the request is from a web browser, the response SHOULD be HTML.
- The suggested definition of "a web browser" is that the Accept header contains "*/*" and the User-Agent header contains (case-insensitive) "mozilla".
- This matches all the common graphical web browsers but not HTTP request libraries or command-line browsers.
When clients are sending a representation to the service (e.g. creating a resource), they MUST specify the format using the [Content-Type](http://tools.ietf.org/html/rfc2616#section-14.17 Content-Type) header.
- If the service does not support the specified Content-Type, it SHOULD return a 406 error.
A X-API-Schemas:
header MUST be present in all responses. The value tells the client the URI where they can find the collection of schemas to understand your responses.
- The URI SHOULD be for the particular API version the request is for.
- If the request is for the root level, it SHOULD be for the latest version.
Escaping the forward slash ("/") character is not required by the JSON specification, but requires consideration.
- When responding as JSON, slashes SHOULD NOT be escaped.
- When responding as HTML, strings in the JSON data portion MUST have forward slashes escaped, typically as "/".
- Failure to do this exposes the page to an XSS attack (using strings that contain "</script>").
Services provide a HTML version of their API by wrapping the JSON response with the snippet below. It includes JavaScript and CSS that pretty-prints the response and displays a bar that provides buttons for operations, actions, sorting, filtering, pagination, etc.
<!DOCTYPE html>
<!-- If you are reading this, there is a good chance you would prefer sending an
"Accept: application/json" header and receiving actual JSON responses. -->
<link rel="stylesheet" type="text/css" href="//cdn.rancher.io/api-ui/1.0.0/ui.css" />
<script src="//cdn.rancher.io/api-ui/1.0.0/ui.js"></script>
<script>
var schemas = "http://url-to-your-api/v1/schemas";
var data = {
/* ... JSON response ... */
};
</script>
There are several other options you can set to configure the UI. The full source is available in rancher/api-ui, see it if you''d like to compile and host it yourself or to learn about the options.
Most APIs that do something useful will need some form of authentication to determine what user is making the request and validate that they should be allowed to make it. A user is identified by a set of credentials called an API Key pair.
An API Key consists of a pair of strings called an access key and secret key. These are randomly generated opaque strings assigned by the service to each API user. The access key is analogous to a username and the secret key to a password.
Services MUST support HTTP Basic authentication. In Basic authentication, the client sends their access key and secret key in the Authorization header. The service then reads these and validates the keys.
Every HTTP response contains a status code and is the primary indicator of the success of a request. Requests that were successfully handled MUST return a response with a 2xx (Success) or 3xx (Redirection) status code. Requests that cannot be processed MUST return a 4xx (Client Error) or 5xx (Server Error) code. It is important that these codes be applied correctly and consistently.
With over 70 HTTP Status Codes defined by IANA, it is difficult to remember all of them. To address this a simple rule must be followed: Use HTTP Status Codes if they apply completely.
If there is any question, use a more generic status, such as a 400 or 500, describing the condition in detail following Error guidelines. For more information on when to use specific codes consult Wikipedia or RFC 2616.
HTTP Status Codes come in five categories that identify their meaning:
- 1xx: Informational - Request was received, more information was returned.
- 2xx: Success - The request was successfully received, understood and accepted.
- 3xx: Redirection - The request must be performed elsewhere to be completed.
- 4xx: Client Error - The request contained invalid information that the client must address. Attempting the same request will result in the same error.
- 5xx: Server Error - The request could not be fulfilled by the server. The same request may succeed (or produce a 4xx error) if attempted later.
At a minimum the following HTTP Status Codes should be well known and used.
Code | Meaning |
---|---|
200 OK | The request was successful |
201 Created | Success, and a new resource has been created. A Location header to the resource SHOULD be included. |
202 Accepted | The request has been received but has not completed. See Asynchronous Actions |
400 Bad Request | The request was malformed in some way. Used for general errors that can be corrected by the client. |
401 Unauthorized | Authentication information was not sent or is invalid. |
403 Forbidden | The authenticated user is not allowed access to the requested resource. |
404 Not Found | These are not the droids you are looking for. |
422 Unprocessable Entity | The request was well-formed (400) and in a supported format (415), but cannot be processed. Typically used for application-specific validation errors. |
500 Internal Server Error | A generic message for an error on the server. The client should try their request again later at which point it may succeed (or produce a 4xx error). |
The remaining HTTP Status Codes will be used under certain specific circumstances.
Code | Meaning |
---|---|
301 Moved Permanently | Permanent redirect. The requested resource has moved permanently and clients should never request this URI again. If in doubt, use a 302. |
302 Found | Temporary redirect. The requested resource has moved and clients should request this URI again for future requests. |
304 Not Modified | The client requested a resource that allows a previous version to be used. No response will be sent. |
405 Method Not Allowed | The requested HTTP method is not allowed for this URL. |
406 Not Acceptable | The service does not support the representation that the client requested. |
409 Conflict | Resource versioning is enabled, and the requested operation conflicts with the current state. |
410 Gone | The resource requested has left. It will not come back. |
413 Entity Too Large | The request body is larger than the service is willing to process. |
414 Request-URI Too Long | The requested URL is longer than is accepted by the service. |
415 Unsupported Media Type | The service does not support the representation type that the client sent. |
418 I'm a teapot | The request attempted to brew coffee with a teapot service that is not compliant with RFC2324. |
503 Service Unavailable | The request couldn't be handled due to maintenance or overload. |
Resource representations MUST have these attributes:
type:
A string that uniquely identifies the kind, or type, of resource this is.- The value MUST be the id of a schema.
SHOULD have, in almost all cases:
id:
A string which uniquely identifies this resource. (See identifier design considerations).links:
A map with links to:self:
The location of the resource itself.
- Other application-specific information attributes
And MAY have, when appropriate:
- Additional
links:
entries for related resources and collections. actions:
A map of action names → the URL that is used to invoke them.
{
"id": "b1b2e7006be",
"type": "file",
"links": { /* see links */ },
"actions": { /* see actions */ },
"name": "ultimate_answer.txt",
"size": 2,
/* ... more application-specific attributes ... */
}
Collections are a special kind of resource. They have a "data" array that contains an array of resources.
Collection representations MUST have:
type:
"collection"resourceType:
The id of the schema for the type of resource this collection holds.data:
An array containing resource representations- This MUST always be present and be an array, even if there are 0 or 1 entries in it.
SHOULD have, in almost all cases:
links:
A map with links to:self:
The location of the resource itself.
And MAY have, when appropriate:
- Additional
links:
entries for related resources and collections. - Additional application-specific attributes (on the collection, as it is itself a resource).
actions:
A map of action names → the URL that can be requested to perform them (on the collection).pagination:
See pagination.sort:
See sorting.filters:
See filtering.createDefaults:
A map of field names → their values.- This values provide a context-specific default value for fields which a client MAY use when creating a new resource in the collection.
createTypes:
See inheritance, below.
{
"type": "collection",
"resourceType": "file",
"links": {
"self": "https://base/v1/files",
/* ... more links ... */
},
"actions": { /* see actions */ },
"pagination": { /* see pagination */ },
"sort": { /* see sorting */ },
"filters": { /* see filtering */ },
"data": [
{/* file resource */},
{/* file resource */},
/* ... more resources ... */
],
/* ... application-specific attributes ... */
}
Collections MAY support multiple resource types through simple inheritance:
- The
resourceType:
SHOULD refer to the 'base' resource type. - Define additional sub-types that have all the same fields and filters as the base type, plus additional info.
- Return a
createTypes:
map in the collection, with a mapping of type ids that can be created → the URL where they SHOULD be POSTed.- If the 'base' type can also be created, it SHOULD be present in the map.
- The URL for each entry MAY be the same as the collection's self link, or it can be different to provide additional flexibility in the service.
"createTypes": {
"file": "https://base/v1/files",
"textFile": "https://base/v1/files",
"imageFile": "https://base/v1/files?image=1",
},
Error responses MUST be a valid resource which gives more information about the error. At a minimum, they MUST include:
type:
"error"status:
the HTTP status code of the responsecode:
A short identifier for the error, suitable for identifying what specific kind of error this is within client code.
Errors SHOULD include:
message:
A short description of the error suitable for a human developer to read.detail:
Additional more detailed information, if available.
The type
, status
, code
, message
and detail
are intended for developer use only and SHOULD NOT be presented directly the end-user. Client SHOULD use the status
and code
to construct their own meaningful message for end-user consumption.
Error responses MUST be returned in the representation format that the client asked for. If the service cannot cater to the representation requested a 406 Status Code should be returned.
- It is rude to send an HTML error page back to a client that made a request for JSON.
- Consider responses that might be returned from application servers and web proxies within your stack.
- Respond with no body if it is not possible to respond in the correct format, or if an error occurs within content negotiation.
{
"type": "error",
"status": 404,
"code": "FolderNotFound",
"message": "The specified folder does not exist",
"folderName": "Documents",
/* ... more application-specific info ... */
}
Links provide a trail for the client to follow to get to related information that is not included in the response.
Clients
- SHOULD use the links provided to navigate resources.
- SHOULD NOT hard-code or string-concatenate URLs (other than the version root).
Services
- SHOULD provide all appropriate links to that a client can get to anything exclusively by navigating resource links.
- SHOULD strive to keep URL layout stable, as clients may save link values and expect to be able to retrieve them later.
- MAY alter URL layout and link values, if necessary.
- This is not considered a breaking change, but the Service SHOULD redirect old URLs to the new ones.
{
"links": {
"self": "https://base/v1/files/b1b2e7006be",
"content": "https://base/v1/files/b1b2e7006be/content",
"folder": "https://base/v1/folders/19c3932wef",
"public": "http://documents.your-files.com/ultimate_answer.txt"
/* ... more links ... */
},
/* ...other attributes... */
}
Link entries SHOULD lead to a valid destination. For example, if you know a resource has no "public" URL because it is not shared, do not include the "public" attribute.
See also: What to link
Link values are strings and MUST be an absolute URL, including the scheme, host, port, and API Version.
- This prevents the client needing resolve relative URLs into absolute ones.
- HTTP compression makes the overhead of transmitting absolute vs relative URLs negligible.
- Port SHOULD be omitted if it is the default for the scheme (e.g. 443 for https).
Handling forward slashes:
- Trailing slashes SHOULD NOT be included on URLs produced by a service.
- SHOULD have no effect on the response if a client sends one.
- Multiple slashes in a URL SHOULD be treated as a single slash.
Each service MUST have a top-level collection called "schemas" which contains "schema" resource for each resource type. The information contained in the schemas provides enough information to build a smart client that knows what actions are available, what fields make up a resource, etc.
The "schemas" collection MUST also have a link to the base URL of the API version.
- This makes the schema collection (https://base/v1/schemas) a single URL that provides a client everything they need to know about the service, similar to a WSDL in a SOAP service.
Each schema resource MUST describe:
id:
: The name of the resource type (e.g. "folder").type:
"schema"links:
:self:
: The URL for this schema (e.g. "https://base/v1/schemas/folder")collection:
If this type can be queried/listed, the URL for doing so (e.g. "http://base/v1/folders").- This SHOULD correspond to the collections that are linked at the version root.
- This link is used by API clients to generate object stubs as the URL to fetch data from.
- This link is used by the HTML UI to provide a dropdown of choices for fields that are of type reference.
resourceFields:
: A map of attribute names available in resources of this type → descriptions of that field (see fields).
And SHOULD describe if applicable:
resourceMethods:
: An array of HTTP methods that are available to some (but not necessarily all) resources of this type.resourceActions:
: A map detailing the actions available to resources of this type (see actions).collectionMethods:
: An array of HTTP methods that are available to a collection of this type.collectionActions:
: A map detailing the actions available to collections of this type (see actions).collectionFields:
: A map detailing the non-standard attribute fields that the collection has. (see fields).collectionFilters:
: A map detailing the filters that are available to collections of this type (see filters).
Each field is defined by a map of properties. Fields MUST have a type:
, which may be a "simple" type:
type | description |
---|---|
"string" | UTF-8 string |
"multiline" | Same as string, but hint to the client that the string may have multiple lines (for HTML: <textarea> ) |
"masked" | Same as string, but hint to the client that they should mask display of the text (for HTML: <input type="password"> ) |
"password" | Same as masked, but hint to the client to additionally offer a password generator |
"float" | JSON uses double-precision IEEE 754 |
"int" | "int"s are just floats in JSON, so the min/max safe values are ± 253. Other clients may use signed 32-bit integers, so be careful with any values that may approach 231. |
"date" | As a string, see dates |
"blob" | Binary data, encoded as a string |
"boolean" | Boolean |
"json" | Arbitrary JSON-parseable string. You should feel bad about using this, but sometimes a field really isn't strongly-typed. |
"version" | Semantic version string. |
Or a non-simple type:
type | description |
---|---|
"enum" | One choice out of an enumeration of possible values |
"reference[schema_name ]" |
The value is the id of a resource of type schema_name |
"schema_name " |
The value is the representation of a resource of type schema_name |
"array[another_type ]" |
The value is an array of entries of type another_type |
"map[another_type ]" |
Key/value pairs. Keys are strings, Values are of type another_type |
Additional metadata SHOULD be provided for each field when applicable:
name | type | description |
---|---|---|
default: |
same as field |
The default value that will be used if none is specified |
unique: |
boolean | If true, this field's value must be unique with its resource type |
nullable: |
boolean | If true, null is a valid value for this field |
create: |
boolean | If true, this the field can be set when creating a resource |
required: |
boolean | If true, this field must be set when creating a resource |
update: |
boolean | If true, this field's value can be changed when updating a resource |
minLength: |
int | For strings and arrays, the minimum length (inclusive) |
maxLength: |
int | For strings and arrays, the maximum length (inclusive) |
min: |
int | For ints and floats, the minimum allowed value (inclusive) |
max: |
int | For ints and floats, the maximum allowed value (inclusive) |
options: |
array[same as field ] |
For enums, the list of possible values |
validChars: |
string | For strings, only these characters are allowed (see character ranges) |
invalidChars: |
string | For strings, these characters are not allowed (see character ranges) |
referenceCollection: |
string | For references, a query URL that can be used to find valid resources of the reference type |
satisfies: |
string | For versions, a semver range which the version must satisfy |
The additional fields above provide enough info for a client to do basic validation of values before submitting them to a service. They are not intended to be completely comprehensive; A service will often have additional restrictions on values that cannot be represented here. If the service is given a value that does not match all of its conditions, it should return a 422 error with enough detail for the client to fix the problem and re-submit the request.
The validChars:
and invalidChars:
are case-sensitive and may contain ranges as in a simple regular expression. For example alphanumeric strings may be represented as "a-zA-Z0-9". Unicode characters should be represented as \uXXXX
or \uXXXXXX
, where X are the hexadecimal representation of the codepoint.
{
"id": "file",
"type": "schema",
"resourceMethods": ["GET","DELETE"],
"resourceFields": {
"name": {"type": "string", "required": true, "create": true, "update": true},
"size": {"type": "int"},
"owner": {"type": "reference[user]"},
"access": {
"type": "enum",
"options": ["public","private","requirepassword"],
"default": "private",
"create": true,
"update": true
},
/* ... more fields ... */
},
"resourceActions": {
"encrypt": {
"input": "cryptInput",
"output": "file",
},
/* ...more actions... */
},
"collectionMethods": ["GET","POST"],
"collectionActions": {
"archive" {
"input": "archiveInput",
"output": "file",
}
/* ...more actions... */
},
"collectionFields": {
"size": {"type": "int"}
},
"collectionFilters": {
"name": {"modifiers": ["ne","prefix","suffix"]},
"date": {"modifiers": ["lt","gt","lte","gte","prefix"]}
"size": {"modifiers": ["lt","gt","lte","gte"] }
"access": {"modifiers": ["eq","ne"], "options": ["public","private","requirepassword"]}
}
}
Actions perform an operation on a resource and (optionally) return a result. They may take an input resource and typically do something that requires application logic beyond what could be done with a simple create, update, or delete operation.
Types that have actions available have an actions:
and/or collectionActions:
map in their schema detailing all the actions that are possible and their input/output schema. For example a file resource may have "share", "encrypt", and "decrypt" actions. See schemas for more information.
Resources have a actions:
attribute that details what actions are available for this particular resource, and what URL is to be accessed to perform them. For example if only one of "encrypt" and "decrypt" are possible for a given file, only the one currently available would appear in the resource:
{
"id": "b1b2e7006be",
"type": "file",
"actions": {
"share": "https://base/v1/files/b1b2e7006be?share",
"encrypt": "https://base/v1/files/b1b2e7006be?encrypt"
},
/* ... other resource attributes ... */
}
Action URLs MUST be considered opaque strings to the client. A simple query parameter is suggested, but services may construct URLs in any format they desire. Examples:
- "https://base/v1/files/b1b2e7006be?encrypt"
- "https://base/v1/files/b1b2e7006be?action=encrypt"
- "https://base/v1/files/b1b2e7006be/encrypt"
- "https://base/v1/actions/encryptFile/b1b2e7006be"
Actions are defined by a map with 2 attributes:
input:
The ID of the schema that the inputs will conform tooutput:
The ID of the schema that the outputs will conform to
Inputs and outputs are resources and MUST have an associated schema. If the action takes no input or has no output, the appropriate attribute SHOULD be omitted.
{
"id": "file",
"type": "schema",
"resourceActions": {
"share": {
"input": "shareInput",
},
"decrypt": {
"input": "cryptInput",
"output": "file",
},
"encrypt": {
"input": "cryptInput",
"output": "file",
},
/* ... other actions ... */
},
/* ... other schema attributes ... */
}
Operations are logical actions with their corresponding HTTP Methods, or Verbs. These operations are listed below and described in detail further below that.
Operation | HTTP Method | Request URL | Description |
---|---|---|---|
Query | GET or HEAD | https://base/v1/{collection_name} | Retrieve a collection of resources |
Read | GET or HEAD | https://base/v1/{collection_name}/{resource id} | Retrieve a single resource |
Create | POST | https://base/v1/{collection_name} | Creates one or more resources |
Update | PUT | https://base/v1/{collection_name}/{resource_id} https://base/v1/{collection_name} |
Change one or more existing resources |
Delete | DELETE | https://base/v1/{collection_name}/{resource_id} https://base/v1/{collection_name} |
Delete one or more existing resources |
Replace | REPLACE | https://base/v1/{collection_name}/{resource_id} https://base/v1/{collection_name} |
Query, Delete and Create one or more resources in one atomic transaction |
Action | POST | https://base/v1/{collection_name}/{resource_id}?{action_name} https://base/v1/{collection_name}?{action_name} |
Perform an action on a resource or collection |
The query operation allows a client to list the resources that are in a collection. Query operations MUST NOT incur side effects.
GET /v1/folders HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"type": "collection",
"resourceType": "folder",
"links": {
"self": "https://base/v1/folders",
/* ... more links ... */
},
"data": [
{
"type": "folder",
"id": "b1b2e7006be",
"name": "Documents",
"links": { /* see links */ },
"actions": { /* see actions */ }
},
/* ... more folder resources ... */
],
"actions": {
"deleteAll": "https://base/v1/folders?deleteAll",
},
"pagination": { /* see pagination */ },
"sort": { /* see sorting */ },
"filters": { /* see filtering */ }
}
Collections that may return a large number of results SHOULD paginate the response.
Pagination SHOULD be controlled by adding a marker={opaque_string}
query string parameter.
- The implementation and meaning of the opaque string is left to the service.
- A MySQL-style numerical offset is simple, but can return the same entry on 2 subsequent pages if the result set changes between requests.
- A marker that specifically identifies the last row returned prevents this.
Services SHOULD allow the client to specify how many results they would like per page with a limit={integer}
query string parameter.
- The service SHOULD enforce (and document) an upper limit on how high this is allowed to go. At least 1000 is suggested.
- Services SHOULD support
limit=0
so that clients can get just the metadata for a collection (sort, filter, etc).
If the collection supports pagination, the response MUST contain a "pagination" object containing:
limit:
The number of items that are returned in each result set.partial:
Whether this result is the complete set (true) or if it's been truncated (false).
If the result has been truncated, it SHOULD also contain:
first:
The URL of a page containing the first resource in the result set (this SHOULD be the same URL, without a marker).previous:
The URL of the page immediately preceding the current result set.next:
The URL of the page immediately following the current result set.
And MAY additionally contain:
last:
The page containing the last resource in the result set.- This may be impractical to implement, depending on your data source.
total:
The total number of items in the result set, if available.- This may also be impractical, but is very useful information to a client if possible.
When the current result set contains the first resource, 'first' and 'previous' SHOULD NOT be returned. When the current result set contains the last resource, 'next' and 'last' SHOULD NOT be returned.
URLs included in this section SHOULD include all query parameters needed to maintain the current sort and filter.
{
/* ... other collection attributes ... */
"pagination": {
"first": "https://base/v1/files",
"previous": "https://base/v1/files?marker=opaque_string1",
"next": "https://base/v1/files?marker=opaque_string2",
"last": "https://base/v1/files?marker=opaque_string3",
"limit": 100,
"total": 342,
"partial": true
},
/* ... other collection attributes ... */
}
Services MAY allow clients to request a sorted list of resources in a collection by adding sort={sort name}
and order={asc or desc}
query string parameters to the request. Clients MAY use these directly when constructing their query, but SHOULD use the links contained in results.
If a collection can be sorted, the response SHOULD contain a sortLinks:
object, a mapping of attributes that can be sorted → the URL to produce that sort.
If the response was sorted by something, it SHOULD contain a sort:
object containing:
name:
The name of the current sort, so that the client knows which column to show as sorted.order:
The direction of the current sort. This SHOULD be either"asc"
or"desc"
.reverse:
The URL for the same sort in the opposite order.
If no sort is specified by the client, services SHOULD either:
- Choose a default, and include the
sort:
attribute as if it were requested - Or return results in an undefined/unstable order, with no
sort:
attribute).
Sorts MUST be "stable". This means they must produce a unique and repeatable ordering such that the same query will always return results in the same order.
- This is necessary for pagination, and is generally what a client would expect.
- For example the "size" of a file is not unique, so the results should be sorted by something else to make it stable.
- A simple way to do this is to always sort by what the user wants, then by ID or unique name.
URLs included in this section SHOULD include all parameters needed to maintain the current filter, but SHOULD NOT include information about pagination. When following a link to a different sort order, the client SHOULD end up back on the first page of results again.
Multi-level sorting is not defined.
{
/* ... other collection attributes ... */
"sort": {
"name": "size",
"order": "asc",
"reverse": "https://base/v1/files?sort=size&order=desc",
},
"sortLinks": {
"name": "https://base/v1/files?sort=name",
"size": "https://base/v1/files?sort=size",
"upload_date: "https://base/v1/files?sort=upload_date",
/* ... more sort options ... */
},
/* ... other collection attributes ... */
}
Services MAY support searching/filtering of the collection. Support for this is indicated by collectionFilters:
in the schema, and the presence of a filters:
map in the collection response.
The client performs a search by starting with the URL for a standard query request. If they've already done a query and want to filter the results, the self:
link of the result can be used instead. The client then adds on query string parameters for {filter name}{"_"+modifier}={value}
.
- The underscore and modifier are optional if the desired modifier is
eq
.name_eq=a
andname=a
are equivalent.
- For example to search for filenames starting with "a" and greater than 1kb, the client would append:
name_prefix=a&size_gt=1024
- To search for non-image files, they could append:
name_notlike=%.jpg&name_notlike=%.png
All special characters in filter values MUST be URL-encoded. Note: "%" characters are shown here for clarity, but would be "%25" in an actual request.
There is no mechanism defined for OR-ing conditions. All filters are AND-ed together.
Collections that support filtering SHOULD have a filter:
map in the response, with an attribute for each field that can be filtered on.
- If any filter was applied for that field, the value should be an array describing the filters.
- If no filters were applied for that field, the value should be null.
{
/* ... other collection attributes ... */
filters: {
name: [ {"modifier": "prefix", "value":"a" } ],
size: [ {"modifier": "gt", "value": 1024} ],
date: null,
access: null
/* ... more filters ... */
}
/* ... other collection attributes ... */
}
{
/* ... other collection attributes ... */
filters: {
name: [
{"modifier": "notlike", "value": "%.jpg"},
{"modifier": "notlike", "value": "%.png"}
],
size: null,
date: null,
access: null
/* ... more filters ... */
}
/* ... other collection attributes ... */
}
In the schema, the value of each filter attribute SHOULD be an object describing:
modifiers:
What modifiers are supported, if anyoptions:
What specific values are allowed, if the attribute is enumerable
Modifiers allow things like "greater than" or "not equal to". The standard modifiers are:
modifier | meaning |
---|---|
"eq" |
equal to (always available, the default if no modifier is specified) |
"ne" |
not equal to |
"lt" |
less than |
"lte" |
less than or equal to |
"gt" |
greater than |
"gte" |
greater than or equal to |
"prefix" |
starts with |
"like" |
matches, as in SQL: • underscore ("_") for exactly one character • percent ("%") for 0 or more characters • "_" for one underscore character •"%" for one percent character. |
"notlike" |
does not match (as above) |
"null" |
value is NULL |
"notnull" |
value is not NULL |
Services MAY define (and document) their own modifiers.
{
/* ... other collection attributes ... */
"collectionFilters": {
"name": {"modifiers": ["eq","ne","prefix","suffix"]},
"date": {"modifiers": ["eq","lt","gt","lte","gte","prefix"]}
"size": {"modifiers": ["eq","lt","gt","lte","gte"] }
"access": {"modifiers": ["eq","ne"], "options": ["public","private","requirepassword"]}
/* ... more filters ... */
},
/* ... other collection attributes ... */
}
The read operation allows a client to get a single resource. Read operations MUST NOT incur side effects.
GETting the base URL of a service SHOULD return a collection of apiVersion
resources. It SHOULD include a link to the latest:
stable version of the API.
Versions that are not the latest MAY have a deprecated: true
flag to indicate support for this version will be removed in the near future.
{
"type": "collection",
"links": {
"self": "https://base/",
"latest": "https://base/v3",
},
"data": [
{
"id": "v1",
"type": "apiVersion",
"deprecated": true,
"links": {
"self": "https://base/v1",
"file": "https://base/v1/files",
"folder": "https://base/v1/folders"
/* ... other collections and resources ... */
}
},
{
"id": "v2",
"type": "apiVersion",
"deprecated": false,
"links": {
"self": "https://base/v2"
"file": "https://base/v2/files",
"folder": "https://base/v2/folders"
/* ... other collections and resources ... */
}
},
{
"id": "v3",
"type": "apiVersion",
"deprecated": false,
"links": {
"self": "https://base/v3",
"file": "https://base/v3/files",
"folder": "https://base/v3/folders"
/* ... other collections and resources ... */
}
}
]
}
GETting the root URL of a particular API version SHOULD return a resource containing links to all the collections and resources that are available. See API Versioning for more info.
GET /v1 HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"id": "v1",
"type": "apiVersion",
"deprecated": true,
"links": {
"self": "https://base/v1",
"file": "https://base/v1/files",
"folder": "https://base/v1/folders"
/* ... other collections and resources ... */
},
}
GETting the URL of a resource SHOULD return the representation of that resource.
GET /v1/files/b1b2e7006be HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"id": "b1b2e7006be",
"type": "file",
/* ... other resource attributes ... */
}
Making a HEAD request to a URL SHOULD return the same response code and header information as a GET of the same URL would, except that it MUST NOT return any message body.
HEAD /v1/files/b1b2e7006be HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
Content-Length: 234
HEAD /v1/files/non-existent-id HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 404 Not Found
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
Content-Length: 123
The create operation allows a client to create a new resource in a collection.
The body of the request MUST contain a representation of the resource to be created. If any of the fields that are required by the schema for this resource are not included or are invalid, a 422 error SHOULD be returned, with information about what specifically is wrong.
If the created resource is immediately available, a 201 Created status code SHOULD be returned, and the body SHOULD be the representation of the new resource. If the resource cannot be created and read immediately, a 202 Accepted status SHOULD be returned instead. At least the resource id should be returned if at all possible.
For a single resource, the response SHOULD include a Location header with the URL for the new resource. This is likely the same as the self:
link in the resource. Note that Location headers MUST always be an absolute URL.
POST /v1/folders HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
{
"name": "Documents"
}
HTTP/1.1 201 Created
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
Location: https://base/v1/folders/ab391827x31
{
"id": "ab391827x31",
"type": "folder",
/* ... more resource attributes ... */
}
Services SHOULD also support create calls which are HTML Form encoded, Content-Type: multipart/form-data
or Content-Type: application/x-www-form-urlencoded
. These are easy for a client to construct with a HTML form or cURL, and also allow for simple uploading of binary data/files. A file management API might allow creating a file resource like:
curl -u access_key:secret_key \
-F name="ultimate_answer.txt" \
-F file="@mydatafile.txt" \
"http://base/v1/folders/ab391827x31"
Which would make a request like:
POST /v1/folders/ab391827x31 HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Length: 307
Content-Type: multipart/form-data; boundary=----------------------------9a41916dd20a
------------------------------9a41916dd20a
Content-Disposition: form-data; name="name"
ultimate_answer.txt
------------------------------9a41916dd20a
Content-Disposition: form-data; name="file"; filename="mydatafile.txt"
Content-Type: text/plain
42
------------------------------9a41916dd20a--
HTTP/1.1 201 Created
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
Location: https://base/v1/files/b1b2e7006be
{
"id": "b1b2e7006be",
"type": "file",
/* ... more resource attributes ... */
}
Services MAY allow multiple resources to be created in one request by accepting an array of representations instead of a single one.
- A Location header SHOULD NOT be returned.
- The response SHOULD be a collection containing the created resources.
- The service MUST ensure that this operation is atomic:
- Either ALL the creates succeed and a 2xx response code is returned,
- or NONE of the changes are committed and a 4xx or 5xx code is returned.
- If this is not possible then the service SHOULD reject requests for multi-resource create with a 406 error.
POST /v1/folders HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
[
{"name": "Documents", /* ... more resource attributes ... */},
{"name": "Pictures", /*... more resource attributes ... */},
/* ... more resource representations ... */
]
HTTP/1.1 201 Created
X-API-Schemas: https://base/v1/schemas
Content-Type: application/json
{
"type": "collection",
"resourceType": "folder",
"data": [
{"name": "Documents", /* ... more resource attributes ... */},
{"name": "Pictures", /* ... more resource attributes ... */},
/* ... more resource representations ... */
]
/* ... more collection attributes ... */
}
When to use a POST over a PUT to create a resource is better described in an example. A POST would be used when the service generates an ID, where a PUT is used when a resource should be created at a specific location:
A POST is used to create a resource that generates an ID:
POST /v1/files HTTP/1.1
A PUT is used to create a resource where the ID is known:
PUT /v1/files/b1b2e7006be HTTP/1.1
The update operation allows a client to modify resources. Update operations MUST be idempotent; making the same update request again should have no effect.
- Any change which does not have this property is an action, not an update.
Clients may submit just the attributes that they wish to change, or the entire representation of a resource. The id:
, and rev:
(if resource versioning is enabled) attributes MUST always be included.
The response SHOULD include the entire updated resource representation(s), even if the request did not. If the resource is a potentially large binary file instead of a JSON document, a 204 status and no response body SHOULD be returned.
PUT /v1/folders/b1b2e7006be HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
{
"id": "b1b2e7006be",
"rev": "8d2e54afd",
"access": "private"
}
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"id": "b1b2e7006be",
"rev": "8d2e54afd",
"name": "Documents",
"access": "private"
}
When updating multiple resources, the service MUST ensure that the operation is atomic.
- Either ALL the changes succeed and a 2xx response code is returned,
- or NONE of the changes are committed and a 4xx or 5xx code is returned.
- If this is not possible, do not allow multiple updates.
- The service SHOULD document and limit the number of resources that may be updated in one request and return a 400 error if the limit is exceeded.
PUT /v1/folders HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
[
{
"id": "b1b2e7006be",
"rev": "8d2e54afd",
"access": "private"
},
{
"id": "d5a80ee7",
"type": "folder",
"rev": "9586f36z",
"name": "Pictures",
"access": "public"
}
]
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"type": "collection",
"resourceType": "folder",
"data": [
{
"id": "b1b2e7006be",
"type": "folder",
"rev": "8d2e54afd",
"name": "Documents",
"access": "private"
},
{
"id": "d5a80ee7",
"type": "folder",
"rev": "9586f36z",
"name": "Pictures",
"access": "public"
}
]
/* ... other collection attributes ... */
}
The delete operation deletes the item given by the request URL.
DELETE /v1/folders/b1b2e7006be HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 204 No Content
When deleting multiple resources, the service MUST ensure that the operation is atomic.
- Either ALL the deletes succeed and a 2xx response code is returned,
- or NONE of the deletes are committed and a 4xx or 5xx code is returned.
- If this is not possible, do not allow multiple deletes.
- The service SHOULD document and limit the number of resources that may be deletes in one request and return a 400 error if the limit is exceeded.
DELETE /v1/folders/ HTTP/1.1
Content-Type: text/json
[
"b1b2e7006be",
"another_folder_id",
/* ... more resource IDs ... */
]
HTTP/1.1 204 No Content
If appropriate for your service, collections MAY support:
- A "truncate" action to delete all the resources in the collection (but not the collection itself).
- A "recursiveDelete" action to truncate and then delete the collection itself.
Services MAY continue to return resources that have been recently deleted in query and read operations. This is a suggestion from RightScale; It is easier for some use cases for the client to explicitly see that a resource has been removed than it is to infer that it is gone because it no longer comes back in query responses.
Deleted resources:
- MUST have a
removed:
or similar attribute that the client can use to differentiate them from resources that still exist. - SHOULD support filtering on that attribute so clients that don't want them can get results that omit removed results.
- SHOULD stop being returned in query results after a documented period of time on the order of a few hours to days.
The replace operation is an atomic combination of the query, delete, and create operations.
- The request URL is treated as a query operation.
- Any matching resources are deleted.
- New resources described in the request body are created.
Services MAY implement replace for any collections where they see fit. It is most useful when updating a many-to-many relationship. For example, associating a list of IP addresses to a firewall rule. Updating this list would require a client UI to:
- Perform a query to get the current associations.
- Diff the result against what the new entries should be.
- Issue delete operation(s) for the ones that need to be removed.
- Issue create operation(s) for the ones that need to be added.
- Somehow resolve being in an inconsistent state if an error occurs partway through this process.
Replace performs all this in one convenient and atomic action for the client.
- The service MUST ensure that this entire operation is atomic.
- Either ALL the changes succeed and a 2xx response code is returned,
- or NONE of the changes are committed and a 4xx or 5xx code is returned.
The response SHOULD return the list of newly created resources, just as if the request were a normal create.
Note: this operation uses a non-standard HTTP method. There's nothing wrong with that, but some clients may not support arbitrary methods.
REPLACE /v1/firewalls/42/allow HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
[
{"ip": "172.16.234.12"},
{"ip": "10.1.2.3"}
]
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"type": "collection",
"data": [
{"id": "1234", "ip": "172.16.234.12"},
{"id": "1235", "ip": "10.1.2.3"},
}
/* ... more collection attributes ... */
}
The action operation allows a client to manipulate a resource or collection in a way that cannot be done with any of the other standard operations. The request MAY include a body with additional information (arguments), and the response MAY contain a response body with result info.
Actions MAY (and generally SHOULD) incur side effects. The actions that may be possible for given a resource type are defined in its schema. The actions:
attribute in a resource specifies which actions are actually available for this particular resource and the URL to request to perform them.
POST /v1/folders/b1b2e7006be?encrypt HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: application/json
{
"password": "purple monkey dishwasher"
}
HTTP/1.1 202 Accepted
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
/* ... updated representation of the resource ... */
}
Resources and collections MAY be nested. For example, folders might be associated to files, and files to a collection of tags:
GET /v1/folders/d5a80ee7/files/b1b2e7006be/tags HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"type": "collection",
"resourceType": "tag",
"data": [
{"type: "tag", /*... more tag attributes ... */},
/* ... more tag resources ... */
]
/* ... more collection attributes ... */
}
Or the content of the files stored might be exposed as a sub-resource:
PUT /v1/files/b1b2e7006be/content HTTP/1.1
Accept: application/json
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
Content-Type: text/plain
Content-Length: 2
42
HTTP/1.1 204 No Content
Content-Type: application/json
GET /v1/files/b1b2e7006be/content HTTP/1.1
Authorization: Basic YWNjZXNzX2tleTpzZWNyZXRfa2V5
HTTP/1.1 200 OK
Content-Type: text/plain
Content-Length: 2
42
Some (or all) resources MAY implement resource versioning. Versioning allows for the client to perform multiple concurrent updates on a resource and resolve conflicts when they occur.
For resources that implement versioning:
- The service MUST return resource version in responses as the
rev:
attribute in query and read operations. - The client MUST include the
rev:
attribute in update, patch, and action operations.- If the revision given does not match the current revision when the request is processed, the service MUST return a 409 error.
- The service MUST change the
rev:
attribute whenever a field in the resource changes.
The base URL that users access your API with SHOULD be of the form:
https://api{-optional-region-code}.{your product domain}/
APIs SHOULD have a separate DNS record from the your web/app, so that the API can be scaled separately from the rest of your stack. The name can always point to the web/app servers if this scale is not yet needed.
All public APIs that require authentication MUST be accessible over HTTPS, authenticated public APIs MUST NOT be accessible over a non-encrypted channel. Unauthenticated public APIs MAY be provided over HTTP, but SHOULD also offer HTTPS.
APIs MUST support more than one version of their implementation. Clients MUST specify the particular version they want, and the application MUST NOT make changes that are not backwards compatible to that version.
Breaking changes should be avoided when possible, but you are eventually going to have to make a breaking change and it is far better to plan for it from the start. The suggested format for the version string is the letter "v" followed by a single integer which increases by one for each revision (e.g. v1, v2, v3, etc.). The version should be places at the first level of the services URI structure The version SHOULD be treated as an opaque string to the client, so any other format MAY be used.
Versions resources MAY also have a "revision" string attribute that changes on each release. This can be useful to help debug issues caused by changes made within a version that are supposed to be compatible with each other.
Outdated versions SHOULD be deprecated and eventually removed from the service. A request for a version that has been removed SHOULD return a 404 error.
Clients MUST be able to make a GET request to the base URL (without a version fragment) to find out what versions are available. This request SHOULD NOT require authentication. Clients SHOULD check the "latest" link periodically and take action if they are no longer using the latest version.
See the root level and individual version read operation for more info.
Interactions that involve longer processing than a single HTTP transaction should wait or for interactions that have reduced urgency may adopt an asynchronous (queued) behavior over a synchronous (blocking) one. A server that accepts a request from a client MUST respond with a HTTP 201 Accepted result and provide a method to retrieve the results in the response body that include a URI of where the results will be and an absolute time that the client should attempt again.
Once an asynchronous action has been fulfilled, the URI provided to the client will return the results as if it had been performed synchronously, including a typical 2xx or 4xx status code and corresponding response body.
Asynchronous actions MAY fail after having been accepted.
Resource Identifiers
- MUST be unique among other resources of the same type.
- MUST consist only of URL-safe characters.
- SHOULD be globally unique, if practical.
- SHOULD be a string. Numbers require care to avoid eventual overflow.
- MAY be represented as base32 for enhanced readability and error resistance.
- SHOULD NOT be an auto-incrementing ID from a database.
- Exposing this value reveals information about the size and growth of the service, and easily allows attackers to guess other valid IDs.
In general something along the lines of a GUID is best if there will be many of the resource or they are dynamically created. A short human-readable string may be appropriate if there is a very limited set of possible resources.
Each resource SHOULD have a single canonical URL and the representation of that resource SHOULD NOT change depending on how the client got to it.
If a file resource is accessible at https://base/v1/files/b1b2e7006be
and as a data item in a collection https://base/v1/folders/d5a80ee7/files
, both representations SHOULD be exactly the same.
- The links inside the resource SHOULD reflect the canonical URL.
- This prevents the service from generating arbitrarily long link URLs that contain unnecessary history of how the client got to where they are.
- It also makes it possible to compare 2 representations and determine if they are in fact the same resource.
GET /v1/files/b1b2e7006be HTTP/1.1
Accept: application/json
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"id": "b1b2e7006be",
"type": "file",
"links": {
"self": "https://base/v1/files/b1b2e7006be",
"schemas": "https://base/v1/schemas",
"content": "https://base/v1/files/b1b2e7006be/content",
"folder": "https://base/v1/folders/19c3932wef",
"public": "http://documents.your-files.com/ultimate_answer.txt"
},
"actions": {
"encrypt": "htps://base/v1/files/b1b2e7006be?encrypt"
},
"name": "ultimate_answer.txt",
/* ... more attributes ... */
}
GET /v1/folders/d5a80ee7/files HTTP/1.1
Accept: application/json
HTTP/1.1 200 OK
Content-Type: application/json
X-API-Schemas: https://base/v1/schemas
{
"type": "collection",
"links": {
"self": "https://base/v1/folders/d5a80ee7/files",
"schemas": "https://base/v1/schemas"
}
/* ... more attributes ... */
"data": [
{
"id": "b1b2e7006be",
"type": "file",
/* the same JSON for file b1b2e7006be, as above */
}
/* ... more resources ... */
]
}
Wherever possible binary (or non-textual) data should not be contained within its represented state, such as JSON or XML. The preferred method is for the representation to contain an absolute URI to the resource.
If it is desired to embed small amounts of information within a representation, it is advised to conform to the Data URI Scheme as found in CSS and HTML. When utilizing this format services MUST use discrete fields in schema to identify when to expect a URI or inline data. The two should NEVER share the same field.
Several keywords are reserved by this standard and have specific meanings. These words MUST ONLY be used for their documented purpose:
- In schemas:
resourceRepresentations:
- In resources:
id:
,type:
,rev:
links:
,actions:
,length:
- In collections:
- (Everything reserved for resources)
filters:
,pagination:
,sort:
,sortLinks:
,createTypes:
,createDefaults:
data:
- In resource and collection
links:
:self:
- In query strings:
- For pagination:
marker
,limit
- For sorting:
sort
,order
- For usability:
_accept
,_format
,_method
- For pagination:
Some additional guidelines:
- Names and attribute keys should be a single word when practical.
- Single-word collection, resource, attribute names SHOULD be all lowercase.
- Multiple words SHOULD be interCaps (also known as camelCase), not dash-separated, under_scored, or TitleCase.
- Resource names SHOULD be singular, not plural.
- These appear in a resources's
type:
and schema'sid:
.
- These appear in a resources's
REST URI conventions are a flexible concept by nature allowing no "wrong" way to generate one. With usability in mind there are two general rules that SHOULD be followed:
- Resource URIs should remain static
- Simple and short is better than long and explicit
Resources can be classified into two categories, singular and complex relationships. Singular resources are those that stand alone without requiring other resources to function. Complex resources on the other hand have relationships with other resources and can be represented in multiple ways. Take for example a users shopping cart, a user
has a basket
, a basket has item
s. A useful set of URIs for resources would be:
/v1/users # List all Users
/v1/users/1234/basket # Get the basket for User 1234
/v1/basket/5678 # Get Basket 5678
/v1/basket/5678/items # List all Items in Basket 5678
/v1/items # List all Items
/v1/items/9012 # Get Item 9012
In other words, it is RECOMMENDED that complex types follow a /resource/id/subresource
model.
See Canonical Links
Services SHOULD do everything they can to enable the clients to take advantage of caching as much as possible.
Read and Query operations SHOULD return a Cache-Control header with values that are appropriate for the information being returned. If the data changes infrequently, allow clients to cache it for a reasonable period of time.
Read operations SHOULD return a HTTP ETag header with an opaque value. The opaque value MUST uniquely describe the entire resource, and change if any part of the resource is changed. This can often be done as a hash of the serialized resource representation, e.g. convert the resource to a JSON string and return a sha-sum of the string.
Query operations SHOULD also return an ETag. The opaque value MUST uniquely describe the entire result set and change if any of the resources or metadata are changed/added/removed.
Clients MAY send an If-None-Match header along with their query or read operation. The service SHOULD compute its response, and if that response has the same ETag value, return a 304 status instead of transmitting the entire response again.
Whenever possible, services SHOULD keep a last modified date and return a Last-Modified header on Read and Query operations.
Clients MAY send an If-Modified-Since header. The service SHOULD return a 304 status instead of transmitting the entire response again if it hasn't changed since the time given.
API Services SHOULD support gzip and deflate compression and deliver responses with an appropriate Content-Encoding header according to the Accept-Encoding requested by the client.
Many clients, frameworks, and web servers limit the length of query strings and the request URL as a whole. Services SHOULD be designed so that clients never have any good reason to make a request for a URL longer than 2048 bytes. Services MAY return a 413 error if a request URL is longer than they wish to handle.
Many services have resources that exist in multiple geographic regions.
To provide the easiest management for the user, APIs SHOULD provide a single unified endpoint that can manage resources in all regions. This endpoint MAY have servers in each region and use geographic load balancing (Geographic DNS, Shortest Cost Network Path, etc.) to direct the requests to the closest region.
In applications where centralized management is not possible, each region MAY have a separate endpoint with a region code identified in the base URL.
Guidelines for creating links:
- Every reference to the
id:
of another resource SHOULD have a corresponding link.- For example if a file resource has a
folderId:
field, there should be afolder:
link.
- For example if a file resource has a
- Links to a single resource SHOULD be singular.
- e.g.
"content": "https://base/v1/files/b1b2e7006be/content"
- e.g.
- Links to a collection SHOULD be plural.
- e.g.
"files": "https://base/v1/folders/d5a80ee7/files"
- e.g.
- Limit your URL namespace as much as possible. The less surface area you have exposed the less there is that might need to change later.
- Path components and query parameter names SHOULD be short, meaningful words in all lowercase, easy for a human to read.
- Services SHOULD NOT change the format or construction of URLs within an API version
- In theory, everyone uses the discoverability features and follows links, so services may change URL formats at any time.
- But some clients will inevitably ignore discoverability and hardcode paths into their code.
- So if a URL needs to be changed, provide a 301/302 redirect or release a new API version.