Mojolicious::Plugin::Authentication example with under

TL;DR

An extension to the example of using Mojolicious::Plugin::Authentication.

In previous post Mojolicious::Plugin::Authentication example we took a look at an example for Mojolicious::Plugin::Authentication. I mean, it’s in the title - no big deal.

The example suggests that for every method that we want to put behind the authentication check we should do the following:

get '/private' => sub ($c) {
    return $c->redirect_to('/login') unless $c->is_user_authenticated;
    return $c->render(template => 'private');
};

i.e. do the check with is_user_authenticated() and act accordingly (e.g. by redirecting to the /login page).

As you might have already thought at this point, this is both verbose and error prone. What if we forget to add that check?

Using a condition

If you just don’t like the verbosity and you’re fine with getting a Not Found error when the user tries to access a private page (i.e. you don’t want to redirect to the /login page), you can just use a condition:

get '/private' => (authenticated => 1 ) => sub ($c) {
    return $c->render(template => 'private');
};

From the application’s point of view, this route does not even exist if the authentication condition is not fulfilled. For this reason, the /private endpoint will yield a Not Found error to unauthenticated users.

This still does not solve our problem with being error-prone though - we might forget to put the conditions. Or we might be lazy enough to not want to repeat it for all private endpoints.

Using under

It is possible to use under for fun and profit to install an intermediate checkpoint and then place all routes in a sub-tree.

Here is an example:

under '/authenticated' => sub ($c) {
   return 1 if $c->is_user_authenticated;
   $c->redirect_to('/login');
   return 0;
};
get what => sub ($c) {
   return $c->render(template => 'private', page => 'authenticated/what');
};
get ever => sub ($c) {
   return $c->render(template => 'private', page => 'authenticated/ever');
};

The intermediate stop is at /authenticated. Here, we check with is_user_authenticated() and return 1 if we are fine - this signals to the system that we passed this intermediate stop and we can move further in the route selection. Otherwise, we set the redirection to the login page and return 0, to signal that the route analysis should stop here.

In [Mojolicious::Lite][] applications, all routes that come after the under are assumed to be… under it. So the get what => ... route is actually translated into a GET to /authenticated/what and so on. The route analysis will arrive here only if the intermediate stop above was successful, so all routes after the under are safe and will trigger a redirection to the /login page if the user is not authenticated.

There is still a slight space for some information leak anyway. Trying to hit a non-existent route (like /authenticated/whatever) would not yield a redirection to the /login page, but a Not Found error. This somehow exposes our endpoints, and we might want to keep them… private.

If this is a concern, you can add the following final route:

get '*' => sub ($c) { return $c->render(status => 404) };

In this way we are putting a catchall that will eventually complain with a Not Found error, but this time only to authenticated users.

The complete example

I’m planning to keep the complete example and possibly expand it if needed, here’s the current status:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
#!/usr/bin/env perl
use FindBin '$Bin';
use lib "$Bin/local/lib/perl5", "$Bin/lib";

use Mojolicious::Lite -signatures;

{
   my %db = (
      foo => {pass => 'FOO', name => 'Foo De Pois'},
      bar => {pass => 'BAZ', name => 'Bar Auangle'},
   );
   sub load_account ($u) { return $db{$u} // undef }
   sub validate ($u, $p) {
      warn "user<$u> pass<$p>\n";
      my $account = load_account($u) or return;
      return $account->{pass} eq $p;
   }
}

app->plugin(
   Authentication => {
      load_user     => sub ($app, $uid) { load_account($uid) },
      validate_user => sub ($c, $u, $p, $e) { validate($u, $p) ? $u : () },
   }
);

app->hook(
   before_render => sub ($c, $args) {
      my $user = $c->is_user_authenticated ? $c->current_user : undef;
      $c->stash(user => $user);
      return $c;
   }
);

get '/' => sub ($c) { $c->render(template => 'index') };

get '/login' => sub ($c) { $c->render(template => 'login') };

post '/login' => sub ($c) {
   my $username = $c->param('username');
   my $password = $c->param('password');
   my $auth_ok = $c->authenticate($username, $password);
   $c->redirect_to($auth_ok ? 'private' : 'login');
   return;
};

get '/private' => sub ($c) {
   return $c->redirect_to('/login') unless $c->is_user_authenticated;
   return $c->render(template => 'private');
};

post '/logout' => sub ($c) {
   $c->logout if $c->is_user_authenticated;
   return $c->redirect_to('/');
};

under '/authenticated' => sub ($c) {
   return 1 if $c->is_user_authenticated;
   $c->redirect_to('/login');
   return 0;
};
get 'what' => sub ($c) {
   return $c->render(template => 'private', page => 'authenticated/what');
};
get 'ever' => sub ($c) {
   return $c->render(template => 'private', page => 'authenticated/ever');
};
get '*' => sub ($c) { return $c->render(status => 404) };

app->start;

__DATA__
@@ layouts/layout.html.ep
<!DOCTYPE html>
<html lang="en">
  <head><title>Whatevah</title></head>
  <body>
    <%= content %>
    <hr>
%= t a => href => '/' => 'Home';
-
%= t a => href => '/login' => 'Login';
-
%= t a => href => '/private' => 'Private';
% if (defined $user) {
<form action="/logout" method="post"
   style="display: inline"
>
-
<button type="submit"
   style="background: none!important;
          border: none;
          padding: 0!important;
          text-decoration: underline;
          cursor: pointer;
          color: #069;">Logout <%= $user->{name} %></button>
</form>
% }
  </body>
</html>

@@ index.html.ep
% layout 'layout';
%= t h1 => 'Index - Free Access'

@@ private.html.ep
% layout 'layout';
%= t h1 => 'Private Stuff - Retricted Access'
Welcome <%= $user->{name} %><br />
%= t em => 'You are in ' . ($c->stash('page') // 'private')

@@ login.html.ep
% layout 'layout';
%= t h1 => 'login'
%= form_for '/login' => (method => 'post') => begin
username: <%= text_field 'username' %>
password: <%= password_field 'password' %>
%= submit_button 'log in' 
%= end

Comments? Octodon, , GitHub, Reddit, or drop me a line!