From Media Type to MicroTypes
Filippos Vasilakis
Web Engineer at Kollegorna
and Introspected REST
Media Types
RFC 2046 & RFC 6838
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
- response format & structure
- request format & structure
- Hypermedia (HATEOAS)
- Capabilities
- pagination, sorting
- filter queries
- aggregation queries
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
A Media Type specifies all our APIs semantics
For example:
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
{
"data": {
"type": "articles",
"id": "1",
"attributes": {
// ... this article's attributes
},
"relationships": {
// ... this article's relationships
}
}
}
{
"meta": {
"copyright": "Copyright 2015 Example Corp.",
"authors": [
"Yehuda Katz",
"Steve Klabnik",
"Dan Gebhardt",
"Tyler Kellen"
]
},
"data": {
// ...
}
}
GET /articles/1?include=author,comments.author
GET /articles?include=author&fields[articles]=title,body&fields[people]=name
GET /articles?sort=-created,title
"links": {
"self": "http://example.com/articles?page[number]=3&page[size]=1",
"first": "http://example.com/articles?page[number]=1&page[size]=1",
"prev": "http://example.com/articles?page[number]=2&page[size]=1",
"next": "http://example.com/articles?page[number]=4&page[size]=1",
"last": "http://example.com/articles?page[number]=13&page[size]=1"
}
HTTP/1.1 422 Unprocessable Entity
Content-Type: application/vnd.api+json
{
"errors": [
{
"status": "422",
"source": { "pointer": "/data/attributes/first-name" },
"title": "Invalid Attribute",
"detail": "First name must contain at least three characters."
}
]
}
JSONAPI Media Type
{
"data": [{
"type": "articles",
"id": "1",
"attributes": {
"title": "JSON API paints my bikeshed!"
},
"links": {
"self": "http://example.com/articles/1"
},
"relationships": {
"author": {
"links": {
"self": "http://example.com/articles/1/relationships/author",
"related": "http://example.com/articles/1/author"
},
"data": { "type": "people", "id": "9" }
}
}
}],
"included": [{
"type": "people",
"id": "9",
"attributes": {
"first-name": "Dan",
"last-name": "Gebhardt",
"twitter": "dgeb"
},
"links": {
"self": "http://example.com/people/9"
}
}]
}
application/vnd.api+json
APIs today are quite powerful (and complex)
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis ; @vasilakisfil
Each Media Type define same concepts in a different way
- Links
- Pagination
- Actions
- Querying
- Embedded resources
- Errors
- Metadata
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis ; @vasilakisfil
doesn't always feel right :(
Need to introduce a small semenatic change in our API ?
No problem! Just add it to the documentation.
The formal way: just create a new Media Type.
No problem! Just add it to the documentation.
Use RFC 6906 (The 'profile' Link Relation Type)
it should never ever alter the semantics Media Type in use
Use RFC 6906 (The 'profile' Link Relation Type)
non self-descriptive content :(
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
"links": {
"self": "http://example.com/articles?page=3&per_page=10",
"first": "http://example.com/articles?page=1&per_page=10",
"prev": "http://example.com/articles?page=2&per_page=10",
"next": "http://example.com/articles?page=4&per_page=10",
"last": "http://example.com/articles?page=13&per_page=10"
}
"links": {
"self": "http://example.com/articles?page={page}&per_page={per_page}",
"first": "http://example.com/articles?page={page}&per_page={per_page}",
"prev": "http://example.com/articles?page={page}&per_page={per_page}",
"next": "http://example.com/articles?page={page}&per_page={per_page}",
"last": "http://example.com/articles?page={page}&per_page={per_page}"
}
Link: <http://example.com/articles?page=3&per_page=10>; rel="self",
<http://example.com/articles?page=1&per_page=10>; rel="first",
<http://example.com/articles?page=2&per_page=10>; rel="prev",
<http://example.com/articles?page=4&per_page=10>; rel="next",
<http://example.com/articles?page=13&per_page=10>; rel="last"
Evolving our pagination
RFC 5988
RFC 6570
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
What if we need
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
- different error messages
- per platform
- RFC 7807 Problem Details for HTTP APIs
-
Different pagination
- pagination on Link header
- URI templates
-
different querying language
- exploit the power of GraphQL
- Different hypermedia
- URI templates in links and actions
- Semantic links
-
More advanced:
- Deprecations
-
Server Side Events
- real time updates
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
Media Types are monoliths
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
- evolve (bad) parts of our API
- support of different classes of clients
- negotiate parts of the API functionality
Solution ?
We cannot
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
also called Features*
small, isolated, reusable, configurable modules that are negotiable and discoverable. MicroTypes compose a Media Type to facilitate the evolvability and extensibility of our API.
microtypes.info
An interface feature is a part of an interface that identifies, describes, and affords a certain kind of interaction across Web APIs.
Ruben Verborgh, Michel Dumontier, A Web API ecosystem through feature-based reuse. CoRR, 2016
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes
Small
- small in scope
- defines only a specific functionality
Isolated
- single responsibility
- no dependencies on other MicroTypes
- dependencies on RFCs and other standards is perfectly valid
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Reusable
- not coupled to a specific Media Type
- solves the problem for broader use
- takes care of edge cases
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
If you use a cool pattern in your API consider it wrapping it in a MicroType
MicroTypes
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Configurable
- people are different, let's embrace it
- let API designer tailor MicroType to her needs
- maximizes reusability
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Configurable
A pagination MicroType
- location of pagination object inside document
- JSON Pointer
- per_page, page and offset query parameter names
- schema.org, JSON-LD etc
- max per_page number (e.g. 50 items per page)
- JSON Schema ?
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Negotiable
- let the client decide the set of MicroTypes based on its capabilities and current state
Discoverable
- inform the client for all available MicroTypes
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
- Small
- Isolated
- Reusable
- Configurable
- Negotiable
- Discoverable
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
Modules!
API
functionalities
monolith interface
API-specific client
API
functionalities
modular interface features
MicroTypes-aware client
defines
implemented as
Top →Down
Bottom →Up
Small, Isolated, Reusable
defines
provides
content taken by Ruben Verborgh, Michel Dumontier, A Web API ecosystem through feature-based reuse. CoRR, 2016, adapted
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
building blocks
a reusable
self-describing
module
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Parent Media Type
JSONAPI Pagination
Strategy
Link Header Pagination Strategy
URI Templates Pagination Strategy
Problem Details for HTTP APIs
(RFC 7807)
JSONAPI
error messages
JSONAPI
Resource(s)/Fields Querying
GraphQL Resource/Fields Querying
Forms and Actions
spec
JSON-LD
main functionalities
response & request format
message format
MicroTypes support
API Version 1.0
API Version 1.1
API Version 1.2
API Version 1.3
API Version 1.4
API Version 1.5
API Version 1.6
API Version 2.0
Deprecated (!)
Removed
Let's not build monoliths, let's not re-invent the wheel either
Deprecations
Introduction | Content Negotiation | Media Types | MicroTypes | Reactive Negotiation & MicroTypes Discovery
Small, Isolated, Reusable
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Configurable
Configuration link(s) can be defined:
- by the parent Media Type
-
OPTIONS /api/pagination
-
- by the MicroType itself
-
GET /api/_microtypes/pagination
-
- a combination
API designer should always be able to decide
ideal
From the client perspective, configuration is called introspection
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Client
URL
API
1. requests the resource
2. returns the most appropriate representation
JSONAPI Pagination
Strategy
Link Header Pagination Strategy
URI Templates Pagination Strategy
Problem Details for HTTP APIs
(RFC 7807)
JSONAPI
error messages
JSONAPI
Resource(s)/Fields Querying
GraphQL Resource/Fields Querying
Forms and Actions
spec
Parent Media Type
(JSONAPI)
Parent Media Type
(HAL)
Accept: application/vnd.api+json; pagination=jsonapi;
Content-Type: application/vnd.api+json; pagination=jsonapi; querying=jsonapi; errors=rfc7807
some content taken by Mozilla's Developer Network, adapted
Accept: application/vnd.api+json;
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi; errors=rfc7807; errors=jsonapi,
application/vnd.hal+json; pagination=jsonapi; querying=jsonapi;
errors=jsonapi; q=0.9
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi;
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi; errors=rfc7807;
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi; errors=rfc7807; errors=jsonapi
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi; errors=rfc7807; errors=jsonapi,
application/vnd.hal+json; pagination=jsonapi; querying=jsonapi;
errors=jsonapi; q=0.9,
application/vnd.hal+json
Media Type parameters
Introduction | Content Negotiation | Media Types | MicroTypes | Reactive Negotiation & MicroTypes Discovery
Negotiable
also called hint-based or server-driven
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Client
URL
API
1. requests the resource
2. returns the most appropriate representation
Discovery
JSONAPI Pagination
Strategy
Link Header Pagination Strategy
URI Templates Pagination Strategy
Problem Details for HTTP APIs
(RFC 7807)
JSONAPI
error messages
JSONAPI
Resource(s)/Fields Querying
GraphQL Resource/Fields Querying
Forms and Actions
spec
Parent Media Type
(JSONAPI)
Parent Media Type
(HAL)
Content-Type: application/vnd.api+json; pagination=jsonapi; querying=jsonapi; errors=rfc7807
some content taken by Mozilla's Developer Network, adapted
Accept: application/vnd.api+json; pagination=jsonapi; querying=graphql;
querying=jsonapi; errors=rfc7807; errors=jsonapi,
application/vnd.hal+json; pagination=jsonapi; querying=jsonapi;
errors=jsonapi; q=0.9,
application/vnd.hal+json
How can clients know?
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
RFC 7231 notes that proactive negotiation has some serious disadvantages
- It is impossible for the server to accurately determine what might be “best” for any given user
- Having the user agent describe its capabilities in every request can be both very inefficient a potential risk to the user’s privacy
- It complicates the implementation
With reactive negotiation (a.k.a., agent-driven negotiation), selection of the best response representation (regardless of the status code) is performed by the user agent after receiving an initial response from the origin server that contains a list of resources for alternative representations
RFC 7231
available since RFC 2068, published in 1997
not a single API uses it (!)
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Client
URL
API
3. request specific representation
Discovery
JSONAPI Pagination
Strategy
Link Header Pagination Strategy
URI Templates Pagination Strategy
Problem Details for HTTP APIs
(RFC 7807)
JSONAPI
error messages
JSONAPI
Resource(s)/Fields Querying
GraphQL Resource/Fields Querying
Forms and Actions
spec
JSON-LD
Parent Media Type
(JSONAPI)
Parent Media Type
(HAL)
2. show available options
1. requests for available
4. return specified representation
Accept: application/vnd.api+json; pagination=uri-templates;
querying=graphql; querying=jsonapi; errors=jsonapi;
errors=rfc7807,
application/vnd.hal+json; pagination=uri-templates;
querying=jsonapi; querying=jsonapi; q=0.9
Content-Type: application/vnd.api+json; pagination=jsonapi; querying=jsonapi; errors=rfc7807
some content taken by Mozilla's Developer Network, adapted
Media Types + MicroTypes
How?
through reactive negotiation
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
There is an HTTP method for that:
HTTP OPTIONS
The OPTIONS method requests information about the communication options available for the target resource, at either the origin server or an intervening intermediary. This method allows a client to determine the options and/or requirements associated with a resource, or the capabilities of a server, without implying a resource action.
RFC 7231
Discovery
through reactive negotiation
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
OPTIONS /api HTTP/1.1
{
"micro-types": {
"application/mt.pagination+jsonapi",
"application/mt.pagination+uri-templates",
"application/mt.pagination+web-linking",
"application/mt.querying+jsonapi",
"application/mt.querying+graphql",
"application/mt.errors+jsonapi",
"application/mt.errors+rfc7807",
"application/mt.linking+json-ld"
"application/mt.actions+siren"
}
}
Link: <https://www.example.com/api/users?microtype=pagination+jsonapi>;
rel="microtype"; name="application/mt.pagination+jsonapi",
<https://www.example.com/api/users?microtype=pagination+uri-templates>;
rel="microtype"; name="application/mt.pagination+uri-templates",
<https://www.example.com/api/users?microtype=pagination+web-linking>;
rel="microtype"; name="application/mt.pagination+web-linking",
<https://www.example.com/api/users?microtype=querying+jsonapi>;
rel="microtype"; name="application/mt.querying+jsonapi",
<https://www.example.com/api/users?microtype=querying+graphql>;
rel="microtype"; name="application/mt.querying+graphql",
<https://www.example.com/api/users?microtype=errors+jsonapi>;
rel="microtype"; name="application/mt.errors+jsonapi",
<https://www.example.com/api/users?microtype=errors+rfc7807">;
rel="microtype"; name="application/mt.errors+rfc7807"",
<https://www.example.com/api/users?microtype=linking+json-ld>;
rel="microtype"; name="application/mt.linking+json-ld",
<https://www.example.com/api/users?microtype=linking+json-ld>;
rel="microtype"; name="application/mt.actions+siren"
HTTP Body
RFC 5988
The method of discovery can be yet another
MicroType
through Proactive Negotiation
Discovery
through reactive negotiation
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
MicroTypes in HTTP
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
- Small
- Isolated
- Reusable
- Configurable
- Negotiable
- Discoverable
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
Modules!
MicroTypes can lead to sustainable API designs
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
- Introduce changes in a standard way
- Composable Media Types
- Discovery and introspection for things that otherwise would go to documentation..
Thank you!
From Media Types to MicroTypes
Filippos Vasilakis | @vasilakisfil
Introduction | Media Types | MicroTypes |MicroTypes in HTTP
From Media Type to MicroTypes
and Introspected REST
From Media Types to MicroTypes
By Filippos Vasilakis
From Media Types to MicroTypes
From Media Types to MicroTypes and Introspected REST
- 2,191