SilverStripe logins via OAuth2, using the PHP League's OAuth2 client
SilverStripe OAuth2-based login functionality, based on the PHP League's OAuth2 client and the SilverStripe OAuth module.
This module adds âLog in with <provider>â buttons to SilverStripeâs default login form, which will authenticate a user with the chosen provider. It also provides configurable access token scopes (or permission levels) and field mapping for storing user data on registration.
This module must be installed with composer. Run composer require bigfork/silverstripe-oauth-login:*
from the command line, and then run a dev/build
.
NOTE: You must first configure your OAuth providers using the configuration options detailed in the SilverStripe OAuth2 module documentation.
To show a login button for a configured provider, you must add them to the new Authenticator
classâ YAML configuration. The configuration has two options avaiable: name
(shown on the âLogin as Xâ button, how this is configured may change in future releases) and scopes
(the desired scopes/permission levels for the access token).
Following on from the Facebook example in the SilverStripe OAuth2 module documentation:
SilverStripe\Core\Injector\Injector:
Bigfork\SilverStripeOAuth\Client\Factory\ProviderFactory:
properties:
providers:
'Facebook': '%$FacebookProvider'
FacebookProvider:
class: 'League\OAuth2\Client\Provider\Facebook'
constructor:
Options:
clientId: '12345678987654321'
clientSecret: 'geisjgoesingoi3h1521onnro12rin'
graphApiVersion: 'v6.0'
Bigfork\SilverStripeOAuth\Client\Authenticator\Authenticator:
providers:
'Facebook': # Matches the key for '$%FacebookProvider' above
name: 'The Facebooks'
scopes: ['email', 'public_profile']
You can customise the look of the login actions for each provider by creating the relevant template, following the naming convention FormAction_OAuth_<ProviderName>
. For example:
<!-- themes/mysite/templates/FormAction_OAuth_Facebook.ss -->
<button type="submit" name="{$Name}" id="{$ID}" class="facebook-login">
Connect with Facebook
</button>
The Bigfork\SilverStripeOAuth\Client\Form\LoginForm
class also provides two extension points, updateFields
and updateActions
for further customisation.
When a provider returns successfully, but returns an error state (for example, when a user chooses to reject the permissions youâre asking for), this module will attempt to return the user to the login screen and display a human-readable error message. As each provider returns error messages in different formats, you may need to add your own error handler in the event that the default handler is unable to show a suitable message. For example:
Bigfork\SilverStripeOAuth\Client\Control\Controller:
error_handlers:
loginerrorhandler:
priority: 10
context: login
class: 'MyLoginErrorHandler'
use Exception;
use League\OAuth2\Client\Provider\AbstractProvider;
use SilverStripe\Control\HTTPRequest;
use SilverStripe\Security\Security;
class MyLoginErrorHandler implements ErrorHandler
{
public function handleError(AbstractProvider $provider, HTTPRequest $request, Exception $exception)
{
$message = $request->getVar('some_error_message_get_var');
if ($message) {
return Security::permissionFailure(null, $message);
}
}
}
Each member that authenticates via an OAuth provider is assigned a âPassportâ - a record which is unique to each OAuth account owner. This allows one SilverStripe account to be linked to multiple OAuth providers, or even linked to multiple individual accounts on the same provider. While both of those are possible, neither is the default behaviour for this module: by default, each new OAuth account will create a new SilverStripe member record. See the multiple providers/accounts and email collisions sections for more information.
When the user registers for the first time with a provider, they will not yet have an associated Member
record in the SilverStripe database. To create that record, this module attempts to copy information from the resource owner returned by the provider.
The default behaviour is to attempt to copy email, first name and surname, though this behaviour can be altered in one of two ways:
GenericMemberMapper
The default mapper (Bigfork\SilverStripeOAuth\Client\Mapper\GenericMemberMapper
) will attempt to copy fields from a mapping array that can be configured in YAML, for example:
Bigfork\SilverStripeOAuth\Client\Mapper\GenericMemberMapper:
mapping:
'Facebook':
'Town': 'Hometown' # Town is the SilverStripe db column, 'Hometown' is in the data returned by Facebook
'Gender': 'Gender'
If more detailed or complex mapping is needed, you can create your own mapper class to handle it. Just implement Bigfork\SilverStripeOAuth\Client\Mapper\MemberMapperInterface
, set up your mapping logic, and then register your new mapper in YAML:
Bigfork\SilverStripeOAuth\Client\Factory\MemberMapperFactory:
mappers:
'Facebook': 'Mysite\MyFacebookMapperClass'
The default behaviour for this module is to treat each OAuth account as a separate SilverStripe account. This is because every website will have bespoke requirements on how multiple accounts should be treated, for example:
It is up to you if, or how, to handle scenarios like this. The typical solution would be to add buttons for âLink X Accountâ that are shown to users in their account once theyâve authenticated initially.
As itâs possible, and likely, for users to have accounts for multiple OAuth providers that each have the same email address, you may encounter an error similar to âCan't overwrite existing member #123 with identical identifier (Email = [email protected])â. This is because the default behaviour for SilverStripe is to ensure that every member record has a unique email address. There are a few different ways to work around this:
Member.unique_identifier_field
config setting to something other than Email
(for example, ID
)GenericMemberMapper
for your providers, but omit the Email
fieldIf youâd like to replace the default authenticator, or change the internal name of the oauth authenticator, you will need to reset the list of authenticators first. You can achieve this with the following approach:
---
Name: app-auth-reset
After:
- '#oauthauthenticator'
---
SilverStripe\Core\Injector\Injector:
SilverStripe\Security\Security:
properties:
Authenticators: null
---
Name: app-auth
After:
- '#app-auth-reset'
---
SilverStripe\Core\Injector\Injector:
SilverStripe\Security\Security:
properties:
Authenticators:
myoauthname: '%$Bigfork\SilverStripeOAuth\Client\Authenticator\Authenticator'
Module rating system helping users find modules that are well supported. For more on how the rating system works visit Module standards
Score not correct? Let us know there is a problem