While a cup of coffee may seem like its two parts of hot water and coffee
grinds, there is much more to it. The same can be said for a well-designed API. A good API does more than just taking input and returning output. It offers a few key items that make it enjoyable and easy to work with.
grinds, there is much more to it. The same can be said for a well-designed API. A good API does more than just taking input and returning output. It offers a few key items that make it enjoyable and easy to work with.
These few key items can sometimes be overlooked or not get the extra polish they deserve which detract from the API. Below we’ll touch upon these and how they can enhance the API experience.
Documentation
The first interaction any developer will have with an API is its documentation. Good documentation is informative and clear which allows developers to quickly start integrating with the API.
Let’s mockup what snippet of what our documentation might look like for a call that allows us to create a coffee bean.
Create Coffee Bean
Description :This will create a coffee bean resource
Endpoint : /coffee-beans
Method: Post
Request Body Schema: application/json
Fields : 
type     string  (required) : Type of coffee bean
region   string  (required) : Region from where coffee is from
roast    string  (required) : Type of roast for this coffee bean
limited  boolean (optional) : If this bean is a limited batch
quantity int     (optional) : Available bag quantity to purchase
Request example
{
  "type"    : "arabica",
  "region"  : "brazil",
  "roast":  : "dark",
  "limited" : true,
  "quantity": 10
}
Response:HTTP code: 201
Content/Type: application/json
Response example:
{
  "coffee_bean": {
    "id": "8ed6b848-150f-490e-903c-e6f6042dbe6b",
    "type": "arabica",
    "region: "brazil",
    "roast": "dark",
    "limited": true,
    "quantity": 10
  }
}Let's dissect each part of the documentation.
First, it starts with a clear name for this API call Create Coffee Bean. Then it is followed up with a description, endpoint, what format the request should be in, and the HTTP method. This gives us enough information to understand the purpose of this call and how we may want to use it.
This is a great starting off point if this is all of our create calls did. If there are available query parameters or specific authentication headers that need to be set these should be documented.
Second, the fields section outlines what should and shouldn’t have to be set in the request call along with their respective types. We also do know 
this request has to be sent as a JSON thanks to the first section of the documentation.
this request has to be sent as a JSON thanks to the first section of the documentation.
Below fields, we show a 
example requestThird, the Response section informs us what kind of response(s) we could expect from the call. In the example above a create call will return a 201 with a 
coffee_beanNow while this response section is clear as to what you can expect for
calls that work what if a call fails? If we had authentication requirements for the create call and those credentials were not supplied then a 401 response could be returned.
calls that work what if a call fails? If we had authentication requirements for the create call and those credentials were not supplied then a 401 response could be returned.
Returning all possible responses good or bad makes it clear what kind of behavior one would expect to encounter with the API…it offers consistency. It also always for developers to building in proper error handling within their integration with the API.
Good URI Design
A URI, Uniform Resource Identifiers, is which resource your API path is modeling.
Let's take our 
coffee-beancoffee-beancoffee-bean/create-coffee-beanBelow is an example of how you could manipulate the coffee-bean resource.
POST   /coffee-beans      Create Coffee Bean
GET    /coffee-beans      List all coffee beans
GET    /coffee-beans/{id} Get specific coffee bean based on ID
Patch  /coffee-beans/{id} Update specific coffee bean based on ID
Delete /coffee-beans/{id} Delete specific coffee bean based on IDYou will notice that our 
coffee-bean-_Consistent Types
No matter what calls we are making on a resource, the types from our requests and responses should be consistent.
Let's revisit the 
create coffee beanintquantitystringAnother consistency that should be achieved is in the returns top-level node. Let's take two calls Get Coffee Bean and List Coffee Beans. These will return two different resources get will return a single coffee bean while the list will return many. So the responses should look like this:
// Get Coffee Bean
{
  "coffee_bean": {}
}
// List Coffee Beans
{
  "coffee_beans": [{},{}]
}Looking at these responses you can easily and quickly determine what resource was called. It also extends itself to easier integrations a developer 
can model an object and easily unmarshal these JSON responses.
can model an object and easily unmarshal these JSON responses.
Pagination
As more and more users interact with your API the amount of data they are storing will also grow. While this may not seem like a problem at first, 
let's imagine that a user has created over 500 different coffee beans
using our API.
let's imagine that a user has created over 500 different coffee beans
using our API.
This can put a heavy load on our API and not to mention it is a very large
payload to return. Having your APIs list calls be paginated will help
alleviate these issues and still allow for users to use your API without having to worry if their List request will time out or take too long.
payload to return. Having your APIs list calls be paginated will help
alleviate these issues and still allow for users to use your API without having to worry if their List request will time out or take too long.
Informative Errors
While we never want an API to not work properly or throw errors, it is bound to happen. This is why an API should return consistent and informative errors.
Here we have a sample error response.
{
  "error": {
    "status_code": 401,
    "message": "unable to authorize request",
    "error_code": "er401auth",
    "resource": "coffee-bean",
   }
}Here our errors will always have a top-level node of 
errorThe benefit to having all errors be returned as JSON like this is as a developer this gives me all the information I may want for logging and error handling on the integration side. With error responses like this, the integration can easily build in error checking and not worry that the API might throw an unknown error.
Final Thoughts
While each of these key items plays a role in having a good API there is
one similarity across all of them. That similarity is consistency. Without consistency in an API, it will be difficult to work with and not offer a stable integration.
one similarity across all of them. That similarity is consistency. Without consistency in an API, it will be difficult to work with and not offer a stable integration.
There are still a few more recommendations I would like to share.
JSON API Specification
If you are designing your resources JSON structure I would recommend following a JSON specification. This will have your APIs JSON structure stay consistent with other public APIs and not throw developers for a loop when you use a JSON schema you came up with.
OpenAPI Specification/Documentation
The OpenAPI Specification (OAS) is a Linux Foundation Collaborative Project aimed to define standards for language-agnostic interface for HTTP 
APIs. If you model your API to a validate OAS yaml/json you can then use this to generate documentation. There are various tools that make this easy the two I recommend would be Swagger or Stoplight.
APIs. If you model your API to a validate OAS yaml/json you can then use this to generate documentation. There are various tools that make this easy the two I recommend would be Swagger or Stoplight.
There are also
- https://www.openapis.org/
- https://github.com/OAI/OpenAPI-Specification
- https://swagger.io
- https://stoplight.io
Also published at https://medium.com/swlh/what-makes-a-good-api-7126d1630f96
