JSONAPI Elixir
A project that will render your data models into JSONAPI Documents and parse/verify JSONAPI query strings.
JSONAPI Support
This library implements version 1.1 of the JSON:API spec.
- Basic JSONAPI Document encoding
- Basic support for compound documents
- Links
- Relationship links
-
Parsing of
sortquery parameter into Ecto Query order_by -
Parsing and limiting of
filterkeywords. - Handling of sparse fieldsets
- Handling of includes
- Handling of pagination
- Handling of top level meta data
Documentation
How to use with Phoenix
Simply add use JSONAPI.View either to the top of your view, or to the web.ex view section and add the
proper functions to your view like so.
defmodule MyApp.PostView do
use JSONAPI.View, type: "posts"
def fields do
[:text, :body, :excerpt]
end
def excerpt(post, _conn) do
String.slice(post.body, 0..5)
end
def meta(data, _conn) do
# this will add meta to each record
# To add meta as a top level property, pass as argument to render function (shown below)
%{meta_text: "meta_#{data[:text]}"}
end
def relationships do
# The post's author will be included by default
[author: {MyApp.UserView, :include},
comments: MyApp.CommentView]
end
end
is an example of a basic view. You can now call render(conn, MyApp.PostView, "show.json", %{data: my_data, meta: meta}) or 'index.json normally.
If you'd like to use this without phoenix simply use the JSONAPI.View and call JSONAPI.Serializer.serialize(MyApp.PostView, data, conn, meta).
Parsing and validating a JSONAPI Request
In your controller you may add
plug JSONAPI.QueryParser,
filter: ~w(name),
sort: ~w(name title inserted_at),
view: PostView
This will add a JSONAPI.Config struct called jsonapi_config to your conn.assigns. If a user tries to
sort, filter, include, or sparse fieldset an invalid field it will raise a plug error that shows the
proper error message.
The config holds the values parsed into things that are easy to pass into an Ecto query, for example
sort=-name will be parsed into sort: [desc: :name] which can be passed directly to the order_by in ecto.
This sort of behavior is consistent for includes. Sparse fieldsets happen in the view using Map.take but when Ecto gets more complex field selection support we will go further to only query the data we need.
You will need to handle filtering yourself, the filter is just a map with key=value.
Camelized or Dasherized Fields
JSONAPI has recommended in the past the use of dashes (-) in place of underscore (_) as a
word separator for document member keys. However, as of JSON API Spec (v1.1), it is now recommended that member names
are camelCased. This library provides various configuration options for maximum flexibility.
Transforming fields requires two steps:
camelCase outgoing fields requires you to set the
:field_transformationconfiguration option. Example:config :jsonapi, field_transformation: :camelize # or dasherizeUnderscoring incoming params (both query and body) requires you add the
JSONAPI.UnderscoreParametersPlug to your API's pipeline. Your pipeline in a Phoenix app might look something like this:pipeline :api do plug(JSONAPI.EnsureSpec) plug(JSONAPI.UnderscoreParameters) end
Spec Enforcement
We include a set of Plugs to make enforcing the JSONAPI spec for requests easy. To add spec enforcement to your application, add JSONAPI.EnsureSpec to your pipeline:
plug JSONAPI.EnsureSpec
Under-the-hood JSONAPI.EnsureSpec relies on three individual plugs:
JSONAPI.ContentTypeNegotiation— Requires theContent-TypeandAcceptheaders are set correctly.JSONAPI.FormatRequired— Verifies that the JSON body matches the expected%{data: %{attributes: attributes}}format.JSONAPI.IdRequired— Confirm theidkey is present in%{data: data}and that it matches the resource'sidin the URI.
Configuration
config :jsonapi,
host: "www.someotherhost.com",
scheme: "https",
field_transformation: :underscore,
remove_links: false,
json_library: Jason- host, scheme. By default these are pulled from the
conn, but may be overridden. - remove_links.
linksdata can optionally be removed from the payload via setting the configuration above totrue. Defaults tofalse. - json_library. Defaults to Jason.
- field_transformation. This option describes how your API's fields word
boundaries are marked. JSON API Spec (v1.1) recommends using camelCase (e.g.
"favoriteColor": blue). If your API uses camelCase fields, set this value to:camelize. JSON:API v1.0 recommended using a dash (e.g."favorite-color": blue). If your API uses dashed fields, set this value to:dasherize. If your API uses underscores (e.g."favorite_color": "red") set to:underscore.
Other
- Feel free to make PR's. I will do my best to respond within a day or two.
- If you want to take one of the TODO items just create an issue or PR and let me know so we avoid duplication.
- If you need help, I am on irc and twitter.
- Example project