Joost Cassee
joost@cassee.net
@jcassee
jcassee.com
Human Environment and Transport Inspectorate
October 19, 2015
URI
http://example.com/ship/9334026
"An entity"
{
  "name": "VOS PROMINENCE",
  "imo": 9334026,
  "type": "cargo"
}Media Type
application/json
Profile
http://example.com/profiles/ship
"It is a ship"
The resource identified by http://example.com/ship/9334026
{
  "name": "VOS PROMINENCE",
  "imo": 9334026,
  "type": "cargo"
}and is represented as application/json by
has profile http://example.com/profiles/ship
A resource identified by a http/https URL
can be manipulated using HTTP methods
GET
PUT
DELETE
PATCH
POST
Retreive a representation
Store a representation
Delete a resource
Update a resource
"Process" a representation
}
idempotent
... for the most part
Expose business logic as state manipulation
href
http://example.com/company/3342
"That entity over there"
{
  "name": "VOS PROMINENCE",
  "imo": 9334026,
  "type": "cargo",
  "ownerHref": "http://example.com/company/3342"
}"That is the owner"
(semantics described by the profile)
Relation
http://example.com/relations/owner
"It has an owner"
The resource identified by http://example.com/ship/9334026
to the resource identified by http://example.com/company/3342
has relation http://example.com/relations/owner
Hypermedia Application Language
application/hal+json
{
  "name": "VOS PROMINENCE",
  "imo": 9334026,
  "type": "cargo",
  ...{
  "name": "VOS PROMINENCE",
  "imo": 9334026,
  "type": "cargo",
  "_links": {
    "self": {
      "href": "http://example.com/ship/9334026"
    },
    "profile": {
      "href": "http://example.com/profiles/ship"
    },
    "http://example.com/relations/owner": {
      "href": "http://example.com/company/3342"
    }
  },
  ...{
  ...
  "_embedded": {
    "http://example.com/relations/owner": {
      "name": "Vroon B.V.",
      "_links": {
        "self": {
          "href": "http://example.com/company/3342"
        },
        "profile": {
          "href": "http://example.com/profiles/company"
        }
      }
    }
  }
}{
  "_links": {
    "rel": { "href": "uri" }
  }
}{
  "property": "uri"
}Hypertext As The Engine Of Application State
"Use links to traverse the API"
"It's not an API, it's a protocol!"
{
  "_links": {
    "http://example.com/relations/owner": {
      "href": "http://example.com/company/3342"
    },
    "http://example.com/relations/crew-search": {
      "href": "http://example.com/ship/9334026/crew{?name,nationality}",
      "templated": true
    }
  }
}
{
  "_links": {
    "http://example.com/relations/owner": {
      "href": "http://example.com/company/3342",
      "type": "application/hal+json",
      "methods": ["GET"]
    }
  }
}give you
require
{
  "_links": {
    "self": { "href": "http://example.com/users/john" },
    "profile": { "href": "http://profiles.example.com/person" }
  }
}{
  "_links": {
    "self": { "href": "http://example.com/clients/4335/contact" },
    "profile": { "href": "http://profiles.example.com/person" }
  }
}{
  ...
  "_links": {
    "self": {
      "href": "http://example.com/ship/9334026"
    },
    "http://example.com/relations/owner": {
      "href": "http://companies.example.com/3342"
    }
  }
}{
"_links": {
  "http://example.com/relations/owner": {
    "href": "http://example.com/company/3342"
  }
}
{
"_links": {
  "http://example.com/relations/owner": {
    "href": "http://companies.example.com/3342"
  }
}
{
  "_links" : {
    "http://example.com/relations/owner" : {
      "href" : "http://example.com/company/3342",
      "deprecation": "http://example.com/docs/deprecated/owner.html"
    },
    "http://example.com/relations/ownership" : {
      "href" : "http://example.com/ship/9334026/ownership/3"
    }
  }
}SUBSCRIBE
id:0
destination:http://example.com/company/3342/shipsMESSAGE
subscription:0
message-id:007
destination:http://example.com/company/3342/ships
content-type:application/hal+json
{
  "total": 5,
  ...