defmodule Ueberauth.Strategy.Patreon do
use Ueberauth.Strategy,
oauth2_module: Ueberauth.Strategy.Patreon.OAuth
alias Ueberauth.Auth.Info
alias Ueberauth.Auth.Credentials
alias Ueberauth.Auth.Extra
@doc """
Handles the initial redirect to the patreon authentication page.
To customize the scope (permissions) that are requested by patreon include
them as part of your url:
"https://www.patreon.com/oauth2/authorize"
"""
def handle_request!(conn) do
scopes = conn.params["scope"] || option(conn, :default_scope)
params =
[scope: scopes]
|> with_state_param(conn)
module = option(conn, :oauth2_module)
redirect!(conn, apply(module, :authorize_url!, [params]))
end
@doc """
Handles the callback from Patreon.
When there is a failure from Patreon the failure is included in the
`ueberauth_failure` struct. Otherwise the information returned from Patreon is
returned in the `Ueberauth.Auth` struct.
"""
def handle_callback!(%Plug.Conn{params: %{"code" => code}} = conn) do
module = option(conn, :oauth2_module)
token = apply(module, :get_token!, [[code: code]])
if token.access_token == nil do
set_errors!(conn, [
error(token.other_params["error"], token.other_params["error_description"])
])
else
fetch_user(conn, token)
end
end
@doc false
def handle_callback!(conn) do
set_errors!(conn, [error("missing_code", "No code received")])
end
@doc """
Cleans up the private area of the connection used for passing the raw Notion
response around during the callback.
"""
def handle_cleanup!(conn) do
conn
|> put_private(:patreon_token, nil)
|> put_private(:patreon_user, nil)
end
@doc """
Fetches the uid field from the Twitch response. This defaults to the option `uid_field` which in-turn defaults to `id`
"""
def uid(conn) do
%{"data" => user} = conn.private.patreon_user
user["id"]
end
@doc """
Includes the credentials from the Patreon response.
"""
def credentials(conn) do
token = conn.private.patreon_token
%Credentials{
token: token.access_token,
token_type: token.token_type,
refresh_token: token.refresh_token,
expires_at: token.expires_at
}
end
@doc """
Fetches the fields to populate the info section of the `Ueberauth.Auth`
struct.
"""
def info(conn) do
%{ "data" => %{
"attributes" => %{
"full_name" => full_name,
"first_name" => first_name,
"last_name" => last_name,
"about" => about,
"image_url" => image_url,
"url" => url,
"email" => email
}
}} = conn.private.patreon_user
%Info{
email: email,
name: full_name,
first_name: first_name,
last_name: last_name,
description: about,
image: image_url,
urls: %{
profile: url
}
}
end
@doc """
Stores the raw information (including the token) obtained from the Patreon
callback.
"""
def extra(conn) do
%Extra{
raw_info: conn.private.patreon_user
}
end
defp fetch_user(conn, token) do
conn = put_private(conn, :patreon_token, token)
case Ueberauth.Strategy.Patreon.OAuth.get(
token.access_token,
"https://www.patreon.com/api/oauth2/v2/identity?fields%5Buser%5D=full_name,email,first_name,last_name,about,image_url,url"
) do
{:ok, %OAuth2.Response{status_code: 401, body: _body}} ->
set_errors!(conn, [error("token", "unauthorized")])
{:ok, %OAuth2.Response{status_code: status_code, body: user}}
when status_code in 200..399 ->
put_private(conn, :patreon_user, user)
{:error, %OAuth2.Error{reason: reason}} ->
set_errors!(conn, [error("OAuth2", reason)])
{:error, %OAuth2.Response{body: %{"message" => reason}}} ->
set_errors!(conn, [error("OAuth2", reason)])
{:error, _} ->
set_errors!(conn, [error("OAuth2", "uknown error")])
end
end
defp option(conn, key) do
Keyword.get(options(conn), key, Keyword.get(default_options(), key))
end
end