Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/hugojosefson/46elks-api-suggestion

Suggestions for how the 46elks.com API can be improved.
https://github.com/hugojosefson/46elks-api-suggestion

Last synced: 29 days ago
JSON representation

Suggestions for how the 46elks.com API can be improved.

Awesome Lists containing this project

README

        

# Suggestion for a new version of the 46elks.com API

The [API for 46elks.com](https://46elks.com/docs/) is very interesting and good
in many ways. There are also ways it could improve to be more predictable, for
example by adopting widely used standards and conventions.

These are some suggestions for how the 46elks.com API can be improved.

## Status of this document

This is work in progress. Not even a draft yet.

In this git repo, I'm also implementing an example API proxy which better
conforms to HTTP, and also to REST with
[hal+json](http://phlyrestfully.readthedocs.org/en/latest/halprimer.html).

[![Build Status](https://travis-ci.org/hugojosefson/46elks-api-suggestion.svg?branch=master)](https://travis-ci.org/hugojosefson/46elks-api-suggestion)

Written by Hugo Josefson .

If you have further suggestions, or like to discuss any change here, feel free
to make a
[Pull Request](https://github.com/hugojosefson/46elks-api-suggestion/pulls) or
file an [Issue](https://github.com/hugojosefson/46elks-api-suggestion/issues) on
this repo!

### TODO

- Implement suggested API as an API proxy.
- ~~Go through existing API docs, and implement everything as I go along,
except callbacks.~~
- ~~Refactor.~~
- ~~Find all callbacks in the current API docs.~~
- ~~Implement proxying callbacks.~~
- Implement client for this API.
- Add IP addresses endpoint.
- Calculate outgoing IP address of current server by bouncing off of
[https://api.ipify.org/](https://api.ipify.org/).
- Get current outgoing IP address of 46elks by sending an SMS to
`+46700000000` with `delivery_report_uri` pointing back to us with a
temporary random code.
- Make sure requests from 46elks to callbacks here are really from 46elks (via
their public IP address).
- Structure this document / repo.

## The suggested changes

### Predictable version identifier in URI

Use `/v2` for the next version of the API, rather than `/a1`, `/a2` or `/b2`.

Reason: Many APIs use `/v1`, `/v2` etc in their URIs to denote the API version.
It is a predictable format.

### Be consistent with JSON vs form POST

Always accept JSON as input, and always respond with JSON.

Reason: This makes it more predictable, and makes the developer not have to keep
track of where JSON goes and where `x-www-form-urlencoded` goes.

### Predictable HTTP methods according to spec, conventions

Use `POST` only for non-idempotent creation of subordinate resources. Make sure
to respond with appropriate status code, and any `Location:` header where
applicable.

Use `PATCH` for replacing only the included properties of a resource.

(Don't use `PUT`. It is for replacing a resource completely with what's in the
request body, setting properties not included to `null`.)

_// TODO more details, examples_

References:

- http://www.restapitutorial.com/lessons/httpmethods.html
- http://restful-api-design.readthedocs.org/en/latest/methods.html#patch-vs-put

Reason: This makes it more predictable, in that HTTP methods are used in the
same way as in many other API's, and according to widely-adopted conventions.

### Support `OPTIONS`, respond with `Allow` header, `405 Method Not Allowed`

Use the standard HTTP mechanisms for specifying in an API which HTTP methods are
allowed and not.

Specifically, respond with `Allow` header which specifies which HTTP methods are
allowed on the current resource. When the resource exists, but the client
attempted to use a disallowed HTTP method, respond `405 Method Not Allowed`
rather than `403 Forbidden` or `404 Not Found`.

Support the HTTP method `OPTIONS` on all resources, so a client can ask the API
for allowed HTTP methods on a resource.

### Use `true`/`false` instead of `"yes"`/`"no"`

In JSON, use `true` and `false` instead of the strings `"yes"` and `"no"`.

Reason: This makes it more predictable, in that if the property sounds like a
boolean flag, it is represented as one too.

Example in current API: http://www.46elks.com/docs#phone-numbers

_With all other suggestions implemented, this specific case of representing a
phone number's active state becomes moot because it is handled by HTTP status
codes. Do keep `true`/`false` representation in mind for future things though._

### Use all lower-case in resource names

Instead of `/Numbers`, `/Calls` and `/SMS`, use `/numbers`, `/calls`, `/sms`.

Reason: Not having to think about whether the resource has an initial upper-case
character or is in all-caps, or something else, makes it more predictable. The
most common way to write identifiers is in all lower-case.

### Split all names, or none of them

For example, instead of `mobilenumber` and `usagelimit`, use `mobile_number` and
`usage_limit`. Just like `sms_uri`.

Reason: Not having to think about whether the resource is written with or
without `_` as wordbreak, makes it more predictable.

### Use `uri` instead of `url`

References:

- http://www.w3.org/TR/uri-clarification/
- http://stackoverflow.com/questions/176264/what-is-the-difference-between-a-uri-a-url-and-a-urn
- https://danielmiessler.com/study/url_vs_uri/
- http://webmasters.stackexchange.com/questions/19101/what-is-the-difference-between-a-uri-and-a-url

### Use relevant HTTP methods and HTTP status codes for phone numbers

Use relevant HTTP methods for requests, and HTTP status codes for responses.

Remove the `"active"` property. This is now represented by response status codes
(`200` vs others), and the `DELETE` method.

Example, for the phone number `+4670000000`:

- `POST /numbers` allocates a new number, returning `201 Created` with a
`Location: https://api.46elks.com/v2/numbers/%2B4670000000` header pointing to
the newly created resource.
- `GET /numbers/%2B4670000000` retrieves information about the phone number.
Response code `200 OK` allocated to current user, `404 Not found` if not
allocated, `403 Forbidden` if allocated to someone else (or `404 Not Found` if
there is a good reason why that fact should not be known to the public).
- `PATCH /numbers/%2B4670000000` with some parameters in the body, reconfigures
an allocated phone number.
- `DELETE /numbers/%2B4670000000` de-allocates a phone number.

### Accept any `2xx` status code when POST'ing sms

When 46elks POST's an sms because it was received at the phone number, it should
accept any `2xx` HTTP status code as delivery confirmation. Specifically, it
must accept `204 No Content` to mean the server has received the sms and chooses
not to respond with an sms of its own.

### Use `application/hal+json`

Supply links to related resources, from any resource where it is applicable.

Let the client use provided links, so they don't need to build the URI's
themselves to each resource, and you don't need to document or set in stone the
actual URI structures.

Provide a common way of representing a resource, and any relevant embedded other
resources.

Do all this using the HAL standard, because it is easier for both developers and
users of an API to refer to an existing standard, rather that rolling their own.

References:

- http://stateless.co/hal_specification.html
- http://phlyrestfully.readthedocs.org/en/latest/halprimer.html

#### `GET /v2/me`

##### Response status

`200 OK`

##### Response body

```json
{
"_links": {
"parent": {"href": "https://api.46elks.com/v2"},
"self": {"href": "https://api.46elks.com/v2/me"},
"numbers": {"href": "https://api.46elks.com/v2/me/numbers"},
"sms": {"href": "https://api.46elks.com/v2/me/sms"},
"calls": {"href": "https://api.46elks.com/v2/me/calls"}
},
...
}
```