Guardian.Permissions (Guardian v2.2.4) View Source

An optional plugin to Guardian to provide permissions for your tokens.

These can be used for any token types since they only work on the claims.

Permissions are set on a per implementation module basis. Each implementation module can have their own sets. Permissions are similar in concept to OAuth2 scopes. They're encoded into a token and the permissions granted last as long as the token does. This makes it unsuitable for highly dynamic permission schemes. They're best left to an application to implement.

For example. (at the time of writing) some of the Facebook permissions are:

  • public_profile
  • user_about_me
  • user_actions.books
  • user_actions.fitness
  • user_actions.music

To create permissions for your application similar to these:

defmodule MyApp.Auth.Token do

  use Guardian, otp_app: :my_app,
                         permissions: %{
                         default: [:public_profile, :user_about_me]
                         user_actions: %{
                           books: 0b1,
                           fitness: 0b100,
                           music: 0b1000,
                         }
                       }

  use Guardian.Permissions, encoding: Guardian.Permissions.BitwiseEncoding
  # Per default permissions will be encoded Bitwise, but other encoders also exist
  #  * Guardian.Permissions.TextEncoding
  #  * Guardian.Permissions.AtomEncoding
  # It is even possible to supply your own encoding module

  # snip

  def build_claims(claims, _resource, opts) do
    claims =
      claims
      |> encode_permissions_into_claims!(Keyword.get(opts, :permissions))
    {:ok, claims}
  end
end

This will take the permission set in the opts at :permissions and put it into the "pems" key of the claims as a map of %{set_name => integer}.

The permissions can be defined as a list (positional value based on index) or a map where the value for each permission is manually provided.

They can be provided either as options to use Guardian or in the config for your implementation module.

Once you have a token, you can interact with it.

# Get the encoded permissions from the claims
found_perms = MyApp.Auth.Token.decode_permissions_from_claims(claims)

# Check if all permissions are present
has_all_these_things? =
  claims
  |> MyApp.Auth.Token.decode_permissions_from_claims
  |> MyApp.Auth.Token.all_permissions?(%{default: [:user_about_me, :public_profile]})

# Checks if any permissions are present
show_any_media_things? =
  claims
  |> MyApp.Auth.Token.decode_permissions_from_claims
  |> MyApp.Auth.Token.any_permissions?(%{user_actions: [:books, :fitness, :music]})

Using with Plug

To use a plug for ensuring permissions you can use the Guardian.Permissions module as part of a Guardian pipeline.

# After a pipeline has setup the implementation module and error handler

# Ensure that both the `public_profile` and `user_actions.books` permissions
# are present in the token
plug Guardian.Permissions, ensure: %{default: [:public_profile], user_actions: [:books]}

# Allow the request to continue when the token contains any of the permission sets specified
plug Guardian.Permissions, one_of: [
  %{default: [:public_profile], user_actions: [:books]},
  %{default: [:public_profile], user_actions: [:music]},
]

# Look for permissions for a token in a different location
plug Guardian.Permissions, key: :impersonate, ensure: %{default: [:public_profile]}

If the token satisfies either the permissions listed in ensure or one of the sets in the one_of key the request will continue. If not, then auth_error callback will be called on the error handler with auth_error(conn, {:unauthorized, reason}, options).

Link to this section Summary

Functions

See Guardian.Permissions.Plug.call/2.

See Guardian.Permissions.Plug.init/1.

Provides an encoded version of all permissions, and all possible future permissions for a permission set.

Link to this section Types

Specs

input_label() :: permission_label()

Specs

input_permissions() :: %{optional(input_label()) => input_set()}

Specs

input_set() :: permission_set() | permission()

Specs

label() :: atom()

Specs

permission() :: pos_integer()

Specs

permission_label() :: String.t() | atom()

Specs

permission_set() ::
  [permission_label(), ...] | %{optional(label()) => permission()}

Specs

plug_option() ::
  {:ensure, permission_set()}
  | {:one_of, [permission_set(), ...]}
  | {:key, atom()}
  | {:module, module()}
  | {:error_handler, module()}

Specs

t() :: %{optional(label()) => permission_set()}

Link to this section Functions

See Guardian.Permissions.Plug.call/2.

See Guardian.Permissions.Plug.init/1.

Provides an encoded version of all permissions, and all possible future permissions for a permission set.