Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/paulanthonywilson/basic_auth

Elixir Plug to easily add HTTP basic authentication to an app
https://github.com/paulanthonywilson/basic_auth

Last synced: 2 months ago
JSON representation

Elixir Plug to easily add HTTP basic authentication to an app

Awesome Lists containing this project

README

        

# DEPRECATED

`Plug.BasicAuth` is now part of [Plug](https://github.com/elixir-plug/plug).

# BasicAuth

This is an Elixir Plug for adding basic authentication into an application.

The plug can be configured to use:

1) Static username and password in application configuration

-OR-

2) Your own custom authentication function

Note that if using option (1), prior to 2.2.1 the library was vulnerable to [timing attacks](https://codahale.com/a-lesson-in-timing-attacks/);
we suggest updating `~> 2.2.2`.

If you are using your own custom authentication function, then you are are on your own.
([Plug.Crypto.secure_compare/2](https://hexdocs.pm/plug/1.5.0-rc.0/Plug.Crypto.html#secure_compare/2) is something that may help you compare
binaries in constant time.)

## How to use

Add the package as a dependency in your Elixir project using something along the lines of:
```elixir
defp deps do
[{:basic_auth, "~> 2.2.2"}]
end
```

Add into the top of a controller, or into a router pipeline a plug declaration like:

```elixir
plug BasicAuth, use_config: {:your_app, :your_config}
```

Where :your_app and :your_config should refer to values in your application config.

In your configuration you can set values directly, eg

```elixir

config :your_app, your_config: [
username: "admin",
password: "simple_password",
realm: "Admin Area"
]
```

All configuration is read at runtime to support using
[REPLACE_OS_VARS](http://michal.muskala.eu/2017/07/30/configuring-elixir-libraries.html#distillerys-replaceosvars)
as part of a release.

or choose to get one (or all) from environment variables, eg

```elixir
config :basic_auth, my_auth_with_system: [
username: {:system, "BASIC_AUTH_USERNAME"},
password: {:system, "BASIC_AUTH_PASSWORD"},
realm: {:system, "BASIC_AUTH_REALM"}
]
```

Alternatively you can provide a custom function to the plug to authenticate the user any way
you want, such as finding the user from a database.

```elixir
plug BasicAuth, callback: &User.find_by_username_and_password/3
```

(or optionally provide a realm)

```elixir
plug BasicAuth, callback: &User.find_by_username_and_password/3, realm: "Area 51"
```

Where :callback is your custom authentication function that takes a conn, username
and a password and returns a conn. Your function must return `Plug.Conn.halt(conn)`
if authentication fails, otherwise you can use `Plug.Conn.assign(conn, :current_user, ...)`
to enhance the conn with variables or session for your controller.

The function must have an arity of 3, and be of the form

```elixir
@spec myfunction(Plug.Conn.t, String.t, String.t) :: Plug.Conn.t
```

It will receive a connection, username, and password.

Easy as that!

### Authenticating with custom response body

If you want a custom unauthorised response body you can configure a `custom_response`:

```elixir
plug BasicAuth,
use_config: {:your_app, :your_config},
custom_response: &YouApp.Helpers.unauthorized_response/1

```

or

```elixir

plug BasicAuth,
callback: &User.find_by_username_and_password/3,
realm: "Area 51",
custom_response: &YouApp.Helpers.unauthorized_response/1
```

Where `:custom_response` is a custom response function that takes a `conn`. For example:

```elixir

def unauthorized_response(conn) do
conn
|> Plug.Conn.put_resp_content_type("application/json")
|> Plug.Conn.send_resp(401, ~s[{"message": "Unauthorized"}])
end

```

### Authenticating only for specific actions

If you're looking to authenticate only for a subset of actions in a controller you can use plug's `when action in` syntax as shown below

```elixir
plug BasicAuth, [use_config: {: your_app, : your_config}] when action in [:edit, :delete]
```

additionally you can exclude specific actions using `not`

```elixir
plug BasicAuth, [use_config: {: your_app, : your_config}] when not action in [:index, :show]
```
## Testing controllers with Basic Auth

If you're storing credentials within configuration files, we can reuse them within our test files
directly using snippets like `Application.get_env(:basic_auth)[:username]`.

### Update Tests to insert a basic authentication header

Any controller that makes use of basic authentication, will need an additional header injected into
the connection in order for your tests to continue to work. The following is a brief snippet of how
to get started. There is a more detailed
[blog post](http://www.cultivatehq.com/posts/add-basic-authentication-to-a-phoenix-application/) that
explains a bit more about what needs to be done.

At the top of my controller test I have something that looks like:

```elixir
@username Application.get_env(:the_app, :basic_auth)[:username]
@password Application.get_env(:the_app, :basic_auth)[:password]

defp using_basic_auth(conn, username, password) do
header_content = "Basic " <> Base.encode64("#{username}:#{password}")
conn |> put_req_header("authorization", header_content)
end
```

Then for any tests, I can simply pipe in this helper method to the connection process:
```elixir
test "GET / successfully renders when basic auth credentials supplied", %{conn: conn} do
conn = conn
|> using_basic_auth(@username, @password)
|> get("/admin/users")

assert html_response(conn, 200) =~ "Users"
end
```

And a test case without basic auth for completeness:
```elixir
test "GET / without basic auth credentials prevents access", %{conn: conn} do
conn = conn
|> get("/admin/users")

assert response(conn, 401) =~ "401 Unauthorized"
end
```