Skip to content

paulhenri-l/resty

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Resty

Build Status Code coverage

Resty aims to be like ActiveResource but for Elixir.

ActiveResource is great and as I do not intend to reinvent the wheel a lot of the concepts found in ActiveResource have just been ported to this library so you should feel kinda right at home.

If you do not know what ActiverResource is, it is like an ORM but instead of using a database it works with REST APIs.

Model classes are mapped to remote REST resources by Active Resource much the same way Active Record maps model classes to database tables. When a request is made to a remote resource, a REST JSON request is generated, transmitted, and the result received and serialized into a usable Ruby object.

-- ActiveResource README

Installation

First, add Resty to your mix.exs dependencies:

def deps do
  [{:resty, "~> 1.0.0"}]
end

and run mix deps.get.

Configuration

Out of the box Resty works without needing you to configure anything. But if you want to change the defaults you can do it in your config.exs file. Configurable options can be found in the documentation.

Usage

Creating a resource

First thing first. In order to use Resty and query an API you'll need a resource

Resources can be created thanks to the Resty.Resource.Base module.

defmodule Post do
  use Resty.Resource.Base

  # On which site is the resource available? This can also be set globally in
  # your config.exs file.
  set_site "https://jsonplaceholder.typicode.com"

  # What is the path to the resource?
  set_resource_path "/posts"

  # What attributes exists on the resource?
  define_attributes [:title, :body, :userId]
end

And now we can do CRUD!

Find

Find will issue a GET request to the resource.

{:ok, post} = Post |> Resty.Repo.find(1)

IO.inspect(post) # %Post{id: 1, title: "The title", body: "The body", userId: 1}

Under the hood

# Request -> GET https://jsonplaceholder.typicode.com/posts/1
# Response -> {"id": 1, "title": "The title", "body": "The body", "userId": 1}

Create

Create will issue a POST request to the resource

{:ok, post} = Post.build(title: "The title", body: "The body", userId: 1) |> Resty.Repo.save()

IO.inspect(post) # %Post{id: 1, title: "The title", body: "The body", userId: 1}

Under the hood

# Request -> POST https://jsonplaceholder.typicode.com/posts
# Request Body -> {"title": "The title", "body": "The body", "userId": 1}
# Response -> {"id": 2, "title": "The title", "body": "The body", "userId": 1}

Update

Update will issue a PUT request to the resource

{:ok, post} = Post |> Resty.Repo.find(1)

{:ok, updated_post} = %{post | title: "updated-title"} |> Resty.Repo.save()

IO.inspect(update_post) # %Post{id: 1, title: "updated-title", body: "The body", userId: 1}

Under the hood

# Request -> PUT https://jsonplaceholder.typicode.com/posts/1
# Request Body -> {"title": "updated-title", "body": "The body", "userId": 1}
# Response -> {"id": 1, "title": "updated-title", "body": "The body", "userId": 1}

Delete

Delete will issue a DELETE request to the resource

{:ok, post} = Post |> Resty.Repo.find(1)
{:ok, true} = post |> Resty.Repo.delete()

Under the hood

# Request -> DELETE https://jsonplaceholder.typicode.com/posts/1
# Response -> 204 No Content

Associations

Here are the associations that are supported by resty. Do note that associations won't be automaticaly saved. If you change their values you'll have to manually save them.

BelongTo

Resty is able to automaticaly resolve your belongs to associations.

defmodule User do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/users"
  define_attributes [:name, :email]
end

defmodule Post do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/posts"
  define_attributes [:title, :body, :userId]

  belongs_to User, :user, :userId
end

{:ok, post} = Post |> Resty.Repo.find(1)

IO.inspect(post.user) # %User{id: 1, email: "[email protected]", name: "Leanne Graham"}

Under the hood

# Request -> GET https://jsonplaceholder.typicode.com/posts/1
# Request -> GET https://jsonplaceholder.typicode.com/users/1
# Response -> {"id": 1, "title": "The title", "body": "The body", "userId": 1}
# Response -> {"id": 1, "email": "[email protected]", "name": "Leanne Graham"}

HasOne

Resty is able to automaticaly resolve your has one associations.

If the association is already in the response body it won't get refetched

defmodule Company do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/users/:user_id/company"
  define_attributes [:name]
end

defmodule User do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/users"
  define_attributes [:name, :email]

  has_one Company, :company, :userId
end

{:ok, user} = User |> Resty.Repo.find(1)

IO.inspect(user.company.name) # Romaguera-Crona

Under the hood

# Request -> GET https://jsonplaceholder.typicode.com/users/1
# Request -> GET https://jsonplaceholder.typicode.com/users/1/company
# Response -> {"id": 1, "email": "[email protected]", "name": "Leanne Graham"}
# Response -> {"name": "Romaguera-Crona"}

HasMany

Resty is able to automaticaly resolve your has many associations.

If the association is already in the response body it won't get refetched

defmodule Comment do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/posts/:post_id/comments"
  define_attributes [:postId, :body]
end

defmodule Post do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/posts"
  define_attributes [:title, :body]

  has_many Comment, :comments, :postId
end

{:ok, post} = Post |> Resty.Repo.find(1)

IO.inspect(user.post.comments) # [%Comment{name: "id labore ex et quam laborum""} | _]

Under the hood

# Request -> GET https://jsonplaceholder.typicode.com/posts/1
# Request -> GET https://jsonplaceholder.typicode.com/posts/1/comments
# Response -> {"userId": 1, "id": 1, "title": "...", "body": "..."}
# Response -> [{"postId": 1, "id": 1, "name": "...", "email": "...", "body": "..."}, {...}]

Authentication

Some API will require you to authenticate your requests. Out of the box Resty support Basic and Bearer auth.

Basic

More informations about this auth strategy can be found here

defmodule Post do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/posts"
  define_attributes [:title, :body, :userId]
  with_auth Resty.Auth.Basic, user: "hello", password: "hola"
end

Bearer token

More informations about this auth strategy can be found here

defmodule Post do
  use Resty.Resource.Base

  set_site "https://jsonplaceholder.typicode.com"
  set_resource_path "/posts"
  define_attributes [:title, :body, :userId]
  with_auth Resty.Auth.Beaer, token: "my-token"
end

Docs

The documentation is available here https://hexdocs.pm/resty

The most important modules are Resty.Repo, Resty.Resource.Base and Resty.Resource.

Contributing

If you have any questions about how to use this library feel free to open an issue :)

If you think that the documentation or the code could be improved open a PR and I'll happily review it!