https://github.com/whitehouse/api-standards
https://github.com/whitehouse/api-standards
Last synced: 2 months ago
JSON representation
- Host: GitHub
- URL: https://github.com/whitehouse/api-standards
- Owner: WhiteHouse
- Archived: true
- Created: 2012-12-19T16:26:08.000Z (almost 13 years ago)
- Default Branch: master
- Last Pushed: 2019-06-08T17:00:51.000Z (over 6 years ago)
- Last Synced: 2024-11-02T13:34:19.590Z (about 1 year ago)
- Size: 585 KB
- Stars: 3,084
- Watchers: 204
- Forks: 901
- Open Issues: 34
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
- apis-tools-development - White House Web API Standards
- APIs-tools-collection - White House Web API Standards
- awesome-api-devtools - White House Web API Standards
README
# White House Web API Standards
* [Guidelines](#guidelines)
* [Pragmatic REST](#pragmatic-rest)
* [RESTful URLs](#restful-urls)
* [HTTP Verbs](#http-verbs)
* [Responses](#responses)
* [Error handling](#error-handling)
* [Versions](#versions)
* [Record limits](#record-limits)
* [Request & Response Examples](#request--response-examples)
* [Mock Responses](#mock-responses)
* [JSONP](#jsonp)
## Guidelines
This document provides guidelines and examples for White House Web APIs, encouraging consistency, maintainability, and best practices across applications. White House APIs aim to balance a truly RESTful API interface with a positive developer experience (DX).
This document borrows heavily from:
* [Designing HTTP Interfaces and RESTful Web Services](https://www.youtube.com/watch?v=zEyg0TnieLg)
* [API Facade Pattern](http://apigee.com/about/resources/ebooks/api-fa%C3%A7ade-pattern), by Brian Mulloy, Apigee
* [Web API Design](http://pages.apigee.com/web-api-design-ebook.html), by Brian Mulloy, Apigee
* [Fielding's Dissertation on REST](http://www.ics.uci.edu/~fielding/pubs/dissertation/top.htm)
## Pragmatic REST
These guidelines aim to support a truly RESTful API. Here are a few exceptions:
* Put the version number of the API in the URL (see examples below). Don’t accept any requests that do not specify a version number.
* Allow users to request formats like JSON or XML like this:
* http://example.gov/api/v1/magazines.json
* http://example.gov/api/v1/magazines.xml
## RESTful URLs
### General guidelines for RESTful URLs
* A URL identifies a resource.
* URLs should include nouns, not verbs.
* Use plural nouns only for consistency (no singular nouns).
* Use HTTP verbs (GET, POST, PUT, DELETE) to operate on the collections and elements.
* You shouldn’t need to go deeper than resource/identifier/resource.
* Put the version number at the base of your URL, for example http://example.com/v1/path/to/resource.
* URL v. header:
* If it changes the logic you write to handle the response, put it in the URL.
* If it doesn’t change the logic for each response, like OAuth info, put it in the header.
* Specify optional fields in a comma separated list.
* Formats should be in the form of api/v2/resource/{id}.json
### Good URL examples
* List of magazines:
* GET http://www.example.gov/api/v1/magazines.json
* Filtering is a query:
* GET http://www.example.gov/api/v1/magazines.json?year=2011&sort=desc
* GET http://www.example.gov/api/v1/magazines.json?topic=economy&year=2011
* A single magazine in JSON format:
* GET http://www.example.gov/api/v1/magazines/1234.json
* All articles in (or belonging to) this magazine:
* GET http://www.example.gov/api/v1/magazines/1234/articles.json
* All articles in this magazine in XML format:
* GET http://example.gov/api/v1/magazines/1234/articles.xml
* Specify optional fields in a comma separated list:
* GET http://www.example.gov/api/v1/magazines/1234.json?fields=title,subtitle,date
* Add a new article to a particular magazine:
* POST http://example.gov/api/v1/magazines/1234/articles
### Bad URL examples
* Non-plural noun:
* http://www.example.gov/magazine
* http://www.example.gov/magazine/1234
* http://www.example.gov/publisher/magazine/1234
* Verb in URL:
* http://www.example.gov/magazine/1234/create
* Filter outside of query string
* http://www.example.gov/magazines/2011/desc
## HTTP Verbs
HTTP verbs, or methods, should be used in compliance with their definitions under the [HTTP/1.1](http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html) standard.
The action taken on the representation will be contextual to the media type being worked on and its current state. Here's an example of how HTTP verbs map to create, read, update, delete operations in a particular context:
| HTTP METHOD | POST | GET | PUT | DELETE |
| ----------- | --------------- | --------- | ----------- | ------ |
| CRUD OP | CREATE | READ | UPDATE | DELETE |
| /dogs | Create new dogs | List dogs | Bulk update | Delete all dogs |
| /dogs/1234 | Error | Show Bo | If exists, update Bo; If not, error | Delete Bo |
(Example from Web API Design, by Brian Mulloy, Apigee.)
## Responses
* No values in keys
* No internal-specific names (e.g. "node" and "taxonomy term")
* Metadata should only contain direct properties of the response set, not properties of the members of the response set
### Good examples
No values in keys:
"tags": [
{"id": "125", "name": "Environment"},
{"id": "834", "name": "Water Quality"}
],
### Bad examples
Values in keys:
"tags": [
{"125": "Environment"},
{"834": "Water Quality"}
],
## Error handling
Error responses should include a common HTTP status code, message for the developer, message for the end-user (when appropriate), internal error code (corresponding to some specific internally determined ID), links where developers can find more info. For example:
{
"status" : 400,
"developerMessage" : "Verbose, plain language description of the problem. Provide developers
suggestions about how to solve their problems here",
"userMessage" : "This is a message that can be passed along to end-users, if needed.",
"errorCode" : "444444",
"moreInfo" : "http://www.example.gov/developer/path/to/help/for/444444,
http://drupal.org/node/444444",
}
Use three simple, common response codes indicating (1) success, (2) failure due to client-side problem, (3) failure due to server-side problem:
* 200 - OK
* 400 - Bad Request
* 500 - Internal Server Error
## Versions
* Never release an API without a version number.
* Versions should be integers, not decimal numbers, prefixed with ‘v’. For example:
* Good: v1, v2, v3
* Bad: v-1.1, v1.2, 1.3
* Maintain APIs at least one version back.
## Record limits
* If no limit is specified, return results with a default limit.
* To get records 51 through 75 do this:
* http://example.gov/magazines?limit=25&offset=50
* offset=50 means, ‘skip the first 50 records’
* limit=25 means, ‘return a maximum of 25 records’
Information about record limits and total available count should also be included in the response. Example:
{
"metadata": {
"resultset": {
"count": 227,
"offset": 25,
"limit": 25
}
},
"results": []
}
## Request & Response Examples
### API Resources
- [GET /magazines](#get-magazines)
- [GET /magazines/[id]](#get-magazinesid)
- [POST /magazines/[id]/articles](#post-magazinesidarticles)
### GET /magazines
Example: http://example.gov/api/v1/magazines.json
Response body:
{
"metadata": {
"resultset": {
"count": 123,
"offset": 0,
"limit": 10
}
},
"results": [
{
"id": "1234",
"type": "magazine",
"title": "Public Water Systems",
"tags": [
{"id": "125", "name": "Environment"},
{"id": "834", "name": "Water Quality"}
],
"created": "1231621302"
},
{
"id": 2351,
"type": "magazine",
"title": "Public Schools",
"tags": [
{"id": "125", "name": "Elementary"},
{"id": "834", "name": "Charter Schools"}
],
"created": "126251302"
}
{
"id": 2351,
"type": "magazine",
"title": "Public Schools",
"tags": [
{"id": "125", "name": "Pre-school"},
],
"created": "126251302"
}
]
}
### GET /magazines/[id]
Example: http://example.gov/api/v1/magazines/[id].json
Response body:
{
"id": "1234",
"type": "magazine",
"title": "Public Water Systems",
"tags": [
{"id": "125", "name": "Environment"},
{"id": "834", "name": "Water Quality"}
],
"created": "1231621302"
}
### POST /magazines/[id]/articles
Example: Create – POST http://example.gov/api/v1/magazines/[id]/articles
Request body:
[
{
"title": "Raising Revenue",
"author_first_name": "Jane",
"author_last_name": "Smith",
"author_email": "jane.smith@example.gov",
"year": "2012",
"month": "August",
"day": "18",
"text": "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Etiam eget ante ut augue scelerisque ornare. Aliquam tempus rhoncus quam vel luctus. Sed scelerisque fermentum fringilla. Suspendisse tincidunt nisl a metus feugiat vitae vestibulum enim vulputate. Quisque vehicula dictum elit, vitae cursus libero auctor sed. Vestibulum fermentum elementum nunc. Proin aliquam erat in turpis vehicula sit amet tristique lorem blandit. Nam augue est, bibendum et ultrices non, interdum in est. Quisque gravida orci lobortis... "
}
]
## Mock Responses
It is suggested that each resource accept a 'mock' parameter on the testing server. Passing this parameter should return a mock data response (bypassing the backend).
Implementing this feature early in development ensures that the API will exhibit consistent behavior, supporting a test driven development methodology.
Note: If the mock parameter is included in a request to the production environment, an error should be raised.
## JSONP
JSONP is easiest explained with an example. Here's one from [StackOverflow](http://stackoverflow.com/questions/2067472/what-is-jsonp-all-about?answertab=votes#tab-top):
> Say you're on domain abc.com, and you want to make a request to domain xyz.com. To do so, you need to cross domain boundaries, a no-no in most of browserland.
> The one item that bypasses this limitation is `` tags. When you use a script tag, the domain limitation is ignored, but under normal circumstances, you can't really DO anything with the results, the script just gets evaluated.
> Enter JSONP. When you make your request to a server that is JSONP enabled, you pass a special parameter that tells the server a little bit about your page. That way, the server is able to nicely wrap up its response in a way that your page can handle.
> For example, say the server expects a parameter called "callback" to enable its JSONP capabilities. Then your request would look like:
> http://www.xyz.com/sample.aspx?callback=mycallback
> Without JSONP, this might return some basic javascript object, like so:
> { foo: 'bar' }
> However, with JSONP, when the server receives the "callback" parameter, it wraps up the result a little differently, returning something like this:
> mycallback({ foo: 'bar' });
> As you can see, it will now invoke the method you specified. So, in your page, you define the callback function:
> mycallback = function(data){
> alert(data.foo);
> };
http://stackoverflow.com/questions/2067472/what-is-jsonp-all-about?answertab=votes#tab-top