NAME

Mojolicious::Plugin::OAuth2::Server - Easier implementation of an OAuth2 Authorization Server / Resource Server with Mojolicious

Build Status Coverage Status

VERSION

0.05

SYNOPSIS

use Mojolicious::Lite;

plugin 'OAuth2::Server' => {
    ... # see CONFIGURATION
};

group {
  # /api - must be authorized
  under '/api' => sub {
    my ( $c ) = @_;

    return 1 if $c->oauth; # must be authorized via oauth

    $c->render( status => 401, text => 'Unauthorized' );
    return undef;
  };

  any '/annoy_friends' => sub { shift->render( text => "Annoyed Friends" ); };
  any '/post_image'    => sub { shift->render( text => "Posted Image" ); };
};

any '/track_location' => sub {
  my ( $c ) = @_;

  my $oauth_details = $c->oauth( 'track_location' )
      || return $c->render( status => 401, text => 'You cannot track location' );

  $c->render( text => "Target acquired: @{[$oauth_details->{user_id}]}" );
};

app->start;

Or full fat app:

use Mojo::Base 'Mojolicious';

...

sub startup {
  my $self = shift;

  ...

  $self->plugin( 'OAuth2::Server' => $oauth2_server_config );
}

Then in your controller:

 sub my_route_name {
   my ( $c ) = @_;

   if ( my $oauth_details = $c->oauth( qw/required scopes/ ) ) {
     ... # do something, user_id, client_id, etc, available in $oauth_details
   } else {
     return $c->render( status => 401, text => 'Unauthorized' );
   }

   ...
 }

DESCRIPTION

This plugin enables you to easily (?) write an OAuth2 Authorization Server (AS) and OAuth2 Resource Server (RS) using Mojolicious. It implements the required flows and checks leaving you to add functions that are necessary, for example, to verify an auth code (AC), access token (AT), etc.

In its simplest form you can call the plugin with just a hashref of known clients and the code will "just work" - however in doing this you will not be able to run a multi process persistent OAuth2 AS/RS as the known ACs and ATs will not be shared between processes and will be lost on a restart.

To use this plugin in a more realistic way you need to at a minimum implement the following functions and pass them to the plugin:

login_resource_owner
confirm_by_resource_owner
verify_client
store_auth_code
verify_auth_code
store_access_token
verify_access_token

These will be explained in more detail below, in "REQUIRED FUNCTIONS", and you can also see the tests and examples included with this distribution. OAuth2 seems needlessly complicated at first, hopefully this plugin will clarify the various steps and simplify the implementation.

Note that OAuth2 requires https, so you need to have the optional Mojolicious dependency required to support it. Run the command below to check if IO::Socket::SSL is installed.

$ mojo version

CONFIGURATION

The plugin takes several configuration options. To use the plugin in a realistic way you need to pass several callbacks, documented in "REQUIRED FUNCTIONS", and marked here with a *

authorize_route

The route that the Client calls to get an authorization code. Defaults to GET /oauth/authorize

access_token_route

The route the the Client calls to get an access token. Defaults to POST /oauth/access_token

auth_code_ttl

The validity period of the generated authorization code in seconds. Defaults to 600 seconds (10 minutes)

access_token_ttl

The validity period of the generated access token in seconds. Defaults to 3600 seconds (1 hour)

clients

A hashref of client details keyed like so:

clients => {
  $client_id => {
    client_secret => $client_secret
    scopes        => {
      eat       => 1,
      drink     => 0,
      sleep     => 1,
    },
  },
},

Note the clients config is not required if you add the verify_client callback, but is necessary for running the plugin in its simplest form (when there are *no* callbacks provided)

login_resource_owner *

A callback that tells the plugin if a Resource Owner (user) is logged in. See "REQUIRED FUNCTIONS".

confirm_by_resource_owner *

A callback that tells the plugin if the Resource Owner allowed or disallowed access to the Resource Server by the Client. See "REQUIRED FUNCTIONS".

verify_client *

A callback that tells the plugin if a Client is know and given the scopes is allowed to ask for an authorization code. See "REQUIRED FUNCTIONS".

store_auth_code *

A callback to store the generated authorization code. See "REQUIRED FUNCTIONS".

verify_auth_code *

A callback to verify an authorization code. See "REQUIRED FUNCTIONS".

store_access_token *

A callback to store generated access / refresh tokens. See "REQUIRED FUNCTIONS".

verify_access_token *

A callback to verify an access token. See "REQUIRED FUNCTIONS".

METHODS

register

Registers the plugin with your app - note that you must pass callbacks for certain functions that the plugin expects to call if you are not using the plugin in its simplest form.

$self->register($app, \%config);

oauth

Checks if there is a valid Authorization: Bearer header with a valid access token and if the access token has the requisite scopes. The scopes are optional:

unless ( my $oauth_details = $c->oauth( @scopes ) ) {
  return $c->render( status => 401, text => 'Unauthorized' );
}

REQUIRED FUNCTIONS

These are the callbacks necessary to use the plugin in a more realistic way, and are required to make the auth code, access token, refresh token, etc available across several processes and persistent.

The examples below use monogodb (a db helper returns a MongoDB::Database object) for the code that would be bespoke to your application - such as finding access codes in the database, and so on. You can refer to the tests in t/ and examples in examples/ in this distribution for how it could be done and to actually play around with the code both in a browser and on the command line.

Also note that the examples below have no logging, you should probably make sure to $c->log->debug (or warn/error) when falling through to the various code paths to make debugging somewhat easier. The examples below don't have logging so the code is shorter/clearer

login_resource_owner

A callback to tell the plugin if the Resource Owner is logged in. It is passed the Mojolicious controller object. It should return 1 if the Resource Owner is logged in, otherwise it should call the redirect_to method on the controller and return 0:

my $resource_owner_logged_in_sub = sub {
  my ( $c ) = @_;

  if ( ! $c->session( 'logged_in' ) ) {
    # we need to redirect back to the /oauth/authorize route after
    # login (with the original params)
    my $uri = join( '?',$c->url_for('current'),$c->url_with->query );
    $c->flash( 'redirect_after_login' => $uri );
    $c->redirect_to( '/login' );
    return 0;
  }

  return 1;
};

Note that you need to pass on the current url (with query) so it can be returned to after the user has logged in.

confirm_by_resource_owner

A callback to tell the plugin if the Resource Owner allowed or denied access to the Resource Server by the Client. It is passed the Mojolicious controller object, the client id, and an array reference of scopes requested by the client.

It should return 1 if access is allowed, 0 if access is not allowed, otherwise it should call the redirect_to method on the controller and return undef:

my $resource_owner_confirm_scopes_sub = sub {
  my ( $c,$client_id,$scopes_ref ) = @_;

  my $is_allowed = $c->flash( "oauth_${client_id}" );

  # if user hasn't yet allowed the client access, or if they denied
  # access last time, we check [again] with the user for access
  if ( ! $is_allowed ) {
    $c->flash( client_id => $client_id );
    $c->flash( scopes    => $scopes_ref );

    # we need to redirect back to the /oauth/authorize route after
    # confirm/deny by resource owner (with the original params)
    my $uri = join( '?',$c->url_for('current'),$c->url_with->query );
    $c->flash( 'redirect_after_login' => $uri );
    $c->redirect_to( '/confirm_scopes' );
  }

  return $is_allowed;
};

Note that you need to pass on the current url (with query) so it can be returned to after the user has confirmed/denied access, and the confirm/deny result is stored in the flash (this could be stored in the user session if you do not want the user to confirm/deny every single time the Client requests access).

verify_client

Reference: http://tools.ietf.org/html/rfc6749#section-4.1.1

A callback to verify if the client asking for an authorization code is known to the Resource Server and allowed to get an authorization code for the passed scopes.

The callback is passed the Mojolicious controller object, the client id, and an array reference of request scopes. The callback should return a list with two elements. The first element is either 1 or 0 to say that the client is allowed or disallowed, the second element should be the error message in the case of the client being disallowed:

my $verify_client_sub = sub {
  my ( $c,$client_id,$scopes_ref ) = @_;

  if (
    my $client = $c->db->get_collection( 'clients' )
      ->find_one({ client_id => $client_id })
  ) {
      foreach my $scope ( @{ $scopes_ref // [] } ) {

        if ( ! exists( $client->{scopes}{$scope} ) ) {
          return ( 0,'invalid_scope' );
        } elsif ( ! $client->{scopes}{$scope} ) {
          return ( 0,'access_denied' );
        }
      }

      return ( 1 );
  }

  return ( 0,'unauthorized_client' );
};

store_auth_code

A callback to allow you to store the generated authorization code. The callback is passed the Mojolicious controller object, the generated auth code, the client id, the auth code validity period in seconds, the Client redirect URI, and a list of the scopes requested by the Client.

You should save the information to your data store, it can then be retrieved by the verify_auth_code callback for verification:

my $store_auth_code_sub = sub {
  my ( $c,$auth_code,$client_id,$expires_in,$uri,@scopes ) = @_;

  my $auth_codes = $c->db->get_collection( 'auth_codes' );

  my $id = $auth_codes->insert({
    auth_code    => $auth_code,
    client_id    => $client_id,
    user_id      => $c->session( 'user_id' ),
    expires      => time + $expires_in,
    redirect_uri => $uri,
    scope        => { map { $_ => 1 } @scopes },
  });

  return;
};

verify_auth_code

Reference: http://tools.ietf.org/html/rfc6749#section-4.1.3

A callback to verify the authorization code passed from the Client to the Authorization Server. The callback is passed the Mojolicious controller object, the client_id, the client_secret, the authorization code, and the redirect uri.

The callback should verify the authorization code using the rules defined in the reference RFC above, and return a list with 3 elements. The first element should be a client identifier (a scalar, or reference) in the case of a valid authorization code or 0 in the case of an invalid authorization code. The second element should be the error message in the case of an invalid authorization code. The third element should be a hash reference of scopes as requested by the client in the original call for an authorization code:

my $verify_auth_code_sub = sub {
  my ( $c,$client_id,$client_secret,$auth_code,$uri ) = @_;

  my $auth_codes      = $c->db->get_collection( 'auth_codes' );
  my $ac              = $auth_codes->find_one({
    client_id => $client_id,
    auth_code => $auth_code,
  });

  my $client = $c->db->get_collection( 'clients' )
    ->find_one({ client_id => $client_id });

  $client || return ( 0,'unauthorized_client' );

  if (
    ! $ac
    or $ac->{verified}
    or ( $uri ne $ac->{redirect_uri} )
    or ( $ac->{expires} <= time )
    or ( $client_secret ne $client->{client_secret} )
  ) {

    if ( $ac->{verified} ) {
      # the auth code has been used before - we must revoke the auth code
      # and access tokens
      $auth_codes->remove({ auth_code => $auth_code });
      $c->db->get_collection( 'access_tokens' )->remove({
        access_token => $ac->{access_token}
      });
    }

    return ( 0,'invalid_grant' );
  }

  # scopes are those that were requested in the authorization request, not
  # those stored in the client (i.e. what the auth request restriced scopes
  # to and not everything the client is capable of)
  my $scope = $ac->{scope};

  $auth_codes->update( $ac,{ verified => 1 } );

  return ( $client_id,undef,$scope );
};

store_access_token

A callback to allow you to store the generated access and refresh tokens. The callback is passed the Mojolicious controller object, the client identifier as returned from the verify_auth_code callback, the authorization code, the access token, the refresh_token, the validity period in seconds, the scope returned from the verify_auth_code callback, and the old refresh token,

Note that the passed authorization code could be undefined, in which case the access token and refresh tokens were requested by the Client by the use of an existing refresh token, which will be passed as the old refresh token variable. In this case you should use the old refresh token to find out the previous access token and revoke the previous access and refresh tokens (this is *not* a hard requirement according to the OAuth spec, but i would recommend it).

The callback does not need to return anything.

You should save the information to your data store, it can then be retrieved by the verify_access_token callback for verification:

my $store_access_token_sub = sub {
  my (
    $c,$client,$auth_code,$access_token,$refresh_token,
    $expires_in,$scope,$old_refresh_token
  ) = @_;

  my $access_tokens  = $c->db->get_collection( 'access_tokens' );
  my $refresh_tokens = $c->db->get_collection( 'refresh_tokens' );

  my $user_id;

  if ( ! defined( $auth_code ) && $old_refresh_token ) {
    # must have generated an access token via refresh token so revoke the old
    # access token and refresh token (also copy required data if missing)
    my $prev_rt = $c->db->get_collection( 'refresh_tokens' )->find_one({
      refresh_token => $old_refresh_token,
    });

    my $prev_at = $c->db->get_collection( 'access_tokens' )->find_one({
      access_token => $prev_rt->{access_token},
    });

    # access tokens can be revoked, whilst refresh tokens can remain so we
    # need to get the data from the refresh token as the access token may
    # no longer exist at the point that the refresh token is used
    $scope //= $prev_rt->{scope};
    $user_id = $prev_rt->{user_id};

    # need to revoke the access token
    $c->db->get_collection( 'access_tokens' )
      ->remove({ access_token => $prev_at->{access_token} });

  } else {
    $user_id = $c->db->get_collection( 'auth_codes' )->find_one({
      auth_code => $auth_code,
    })->{user_id};
  }

  if ( ref( $client ) ) {
    $scope  = $client->{scope};
    $client = $client->{client_id};
  }

  # if the client has en existing refresh token we need to revoke it
  $refresh_tokens->remove({ client_id => $client, user_id => $user_id });

  $access_tokens->insert({
    access_token  => $access_token,
    scope         => $scope,
    expires       => time + $expires_in,
    refresh_token => $refresh_token,
    client_id     => $client,
    user_id       => $user_id,
  });

  $refresh_tokens->insert({
    refresh_token => $refresh_token,
    access_token  => $access_token,
    scope         => $scope,
    client_id     => $client,
    user_id       => $user_id,
  });

  return;
};

verify_access_token

Reference: http://tools.ietf.org/html/rfc6749#section-7

A callback to verify the access token. The callback is passed the Mojolicious controller object, the access token, and an optional reference to a list of the scopes. Note that the access token could be the refresh token, as this method is also called when the Client uses the refresh token to get a new access token.

The callback should verify the access code using the rules defined in the reference RFC above, and return false if the access token is not valid otherwise it should return something useful if the access token is valid - since this method is called by the call to $c->oauth you probably need to return a hash of details that the access token relates to (client id, user id, etc)

my $verify_access_token_sub = sub {
  my ( $c,$access_token,$scopes_ref ) = @_;

  if (
    my $rt = $c->db->get_collection( 'refresh_tokens' )->find_one({
      refresh_token => $access_token
    })
  ) {

    if ( $scopes_ref ) {
      foreach my $scope ( @{ $scopes_ref // [] } ) {
        if ( ! exists( $rt->{scope}{$scope} ) or ! $rt->{scope}{$scope} ) {
          return 0;
        }
      }
    }

    # $rt contains client_id, user_id, etc
    return $rt;
  }
  elsif (
    my $at = $c->db->get_collection( 'access_tokens' )->find_one({
      access_token => $access_token,
    })
  ) {

    if ( $at->{expires} <= time ) {
      # need to revoke the access token
      $c->db->get_collection( 'access_tokens' )
        ->remove({ access_token => $access_token });

      return 0;
    } elsif ( $scopes_ref ) {

      foreach my $scope ( @{ $scopes_ref // [] } ) {
        if ( ! exists( $at->{scope}{$scope} ) or ! $at->{scope}{$scope} ) {
          return 0;
        }
      }

    }

    # $at contains client_id, user_id, etc
    return $at;
  }

  return 0;
};

PUTTING IT ALL TOGETHER

Having defined the above callbacks, customized to your app/data store/etc, you can configuration the plugin:

$self->plugin(
  'OAuth::Server' => {
    login_resource_owner      => $resource_owner_logged_in_sub,
    confirm_by_resource_owner => $resource_owner_confirm_scopes_sub,
    verify_client             => $verify_client_sub,
    store_auth_code           => $store_auth_code_sub,
    verify_auth_code          => $verify_auth_code_sub,
    store_access_token        => $store_access_token_sub,
    verify_access_token       => $verify_access_token_sub,
  }
);

This will make the /oauth/authorize and /oauth/access_token routes available in your app, which will call the above functions in the correct order. The helper oauth also becomes available to call in various controllers, templates, etc:

$c->oauth( 'post_image' ) or return $c->render( status => 401 );

REFERENCES

SEE ALSO

Mojolicious::Plugin::OAuth2 - A client side OAuth2 Mojolicious plugin

AUTHOR

Lee Johnson - leejo@cpan.org

LICENSE

This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself. If you would like to contribute documentation or file a bug report then please raise an issue / pull request:

https://github.com/leejo/mojolicious-plugin-oauth2-server