---
title: A Mozilla Persona Plugin for CKAN
subtitle: Login to CKAN sites without creating an account, using Mozilla Persona.
tags: CKAN
redirect_from:
- /post/ckanext-persona/
---
{% include wideimg.html src="/assets/images/persona.png" alt="Logging into CKAN using Persona" title="Logging into CKAN using Persona" figcaption="Logging into CKAN using Persona" %}
[ckanext-persona]({{ page.link }}) is a [CKAN](http://ckan.org/) plugin that
lets users login to CKAN sites using [Mozilla
Persona](http://www.mozilla.org/en-US/persona/).
The Persona plugin adds a *login using your email address* option to CKAN's
login and register pages. As you can see in the screenshot above, CKAN's
traditional username and password login is still available as well.
When a user clicks on the *Email* button it opens a Persona sign in page in a
popup window. After signing into Persona the popup window disappears and the
user is taken back to CKAN and logged in.
Here's a video of the login process:
This was quite a fun plugin to develop, it was one of the first plugins to use
[CKAN's IAuthenticator plugin interface](http://docs.ckan.org/en/latest/extensions/plugin-interfaces.html#ckan.plugins.interfaces.IAuthenticator)
and it revealed some interesting limitations that could be addressed in CKAN.
The way it works is:
{% include wideimg.html src="/assets/images/persona_signin_1.png" alt="Logging into CKAN using Persona" title="Logging into CKAN using Persona" figcaption="Logging into CKAN using Persona" %}
1. The user browses to the login page of the CKAN website and clicks on the
*Email* button. The button opens the Persona sign-in page in a popup window.
2. The user signs in to Persona using their email address, verifying that they
are the owner of the email address. The popup window closes, returning the
user to the CKAN site that they came from.
3. Persona sends an *Identity Assertion Certificate* to CKAN, asserting that
the user owns the email address. Some JavaScript belonging to the Persona
plugin receives this assertion from Persona and posts it to CKAN's login
URL.
4. CKAN's Python code on the server receives a request to its `/login` URL with
the Identity Assertion Certificate in the post parameters. CKAN must now
handle this request and log the user in.
When a request is made to CKAN's `/login` URL CKAN iterates over the active
`IAuthenticator` plugins and calls the `login()` method of each, to see if any
of them can handle this login request. If no plugin logs in the user, CKAN
will fall back to its default username and password login handler.
After finding the Identity Assertion Certificate in the post params and
verifying it, the Persona plugin's `login()` method handles the login request
as follows:
{% include wideimg.html src="/assets/images/persona_signin_2.png" alt="Logging into CKAN using Persona" title="Logging into CKAN using Persona" figcaption="Logging into CKAN using Persona" %}
1. It searches CKAN's database for an existing user
account with the verified email address. There are three possible responses
from the database:
a. There is one existing CKAN user with the email address
(this is what's shown in the video above.)
b. There is no existing user with the email address, so the plugin creates a
new account with this email address and logs the user into it
(this is done silently - the user isn't bothered with the fact that a new
account is created for them on the CKAN site).
c. The CKAN site has multiple users with the same email address.
In this case the plugin shows the user
a list of the user accounts and asks them to choose which one they want
to login to.
(Note: this page isn't implemented yet!)
2. Whichever route we took, we end up with one CKAN account that matches the
Persona email address that was used. The plugin logs the user into the
account by saving the email address and username of the account in CKAN's
Beaker session. (The session is a file on the server that keeps track of
the logged-in user.)
3. Finally, the plugin redirects the user's browser to the dashboard page of
the account they've just logged in to.
On every page load, CKAN calls the `identify()` method of every active
`IAuthenticator` plugin to see if any of them can find out which user is logged
in. If no plugin identifies the logged-in user, CKAN searches for a cookie
created by its default username and password login.
The Persona plugin's `identify()` method will use user name it saved in the
Beaker session to identify the user. This is how the user stays logged-in as
they move from page to page on the site.
Finally, when the user hits the logout button CKAN calls the Persona plugin's
`logout()` method, which deletes the cookie that the plugin created so that its
`identify()` method will no longer find a logged-in user when the next page is
loaded.
## Todo
A few of things still need to be implemented:
* When creating new user accounts, the Persona plugin has to generate a user
name. Currently it just uses a UUID, and the user can change their name in
CKAN once they're logged in.
A nicer solution would be to generate a username based on the email address.
`bob@gmail.com` would get the username `bob`, of if `bob` is already taken
then something like `bob54` (random number appended).
Before being logged in the user should be taken to a *Choose your username*
form where they can enter the username they want, and the form is pre-filled
with the generated username as a suggestion.
When the user enters a username into this form and submits it, it needs to be
checked to make sure that it's a valid and untaken username, and then the
user account can be created and the user logged in.
* When creating new user accounts, the Persona plugin also has to generate a
password, because CKAN currently requires a password for the new account.
This password is never shown to the user so they can't use it to login using
the traditional username and password route. They can only login using
Persona. The generated password is nothing but a potential security issue.
This needs to be fixed in CKAN by allowing user accounts without passwords,
that cannot be logged into the traditional way. See the issue here:
.
* Logging in when there are multiple user accounts with the same email address
isn't implemented yet (CKAN just crashes). This feature will actually
probably be quite handy when done, e.g. for developers who have multiple
accounts on the same site for testing (a normal account, a sysadmin account,
etc).
Other things that still need to be done include JavaScript-less
login with Persona, [CSRF protection](https://developer.mozilla.org/en-US/Persona/Security_Considerations),
tests, a nicer user experience when the Persona verification fails, and
enabling CKAN accounts to have multiple email addresses and
[verifying those addresses with Persona](https://developer.mozilla.org/en-US/Persona/The_implementor_s_guide/Adding_extra_email_addresses_with_Persona).
[Mozilla recommend using Selenium](https://developer.mozilla.org/en-US/Persona/The_implementor_s_guide/Testing?redirectlocale=en-US&redirectslug=Persona%2FThe_implementor_s_guide%2FTesting)
to test logging in with Persona. We've been planning to use Selenium for new
frontend tests in CKAN, so once we get going with that maybe we can write the
ckanext-persona tests in the same way.
To test ckanext-persona, see the [ckanext-persona website](https://developer.mozilla.org/en-US/Persona/The_implementor_s_guide/Adding_extra_email_addresses_with_Persona)
for installation instructions.