We recently had our second Incuna Hack Day where Charlie and I made the decision to start breaking up the main internal site, our venerable Dashboard. It was well on its way to becoming a monolithic beast and the only thing that had stopped me breaking it up before was an easy way to add Single Sign On to the many apps it would become.

Enter Django Social Auth. A cover-all-the-bases authentication app that provides backends for pretty much every service you can think of.

This write-up is based on my experience setting up a few internal apps so it’s fairly opinionated towards that goal, but hopefully it’s still adaptable to other situations!

Setup your App in Google’s API Console Link to heading

In Google’s API Console create a new project.

Create Project

You don’t need to turn on any extra services, so go directly to API Access and hit the giant blue button to get started.

Create OAuth 2.0 Client ID

Enter your Product name and the URL to a logo if you have one. These are the details users will see when they authenticate via Google. Click Next.

Now set up the credentials for your application (this is per environment due to the redirect URI). The example below is for development, but only the hostname needs to change between environments.

Enter Local Hostname

Click more options to set the callback path. The default Social Auth URL for this is /complete/google-oauth2/.

Enter your redirect URI

Click Create client id and grab your Client ID/Client secret combo for the next step.

Setup Django Social Auth Link to heading

Add social_auth to your INSTALLED_APPS and the other settings below:

AUTHENTICATION_BACKENDS = (
'social_auth.backends.google.GoogleOAuth2Backend',
'django.contrib.auth.backends.ModelBackend',
)
LOGIN_REDIRECT_URL = '/'
GOOGLE_OAUTH2_CLIENT_ID = os.environ['GOOGLE_OAUTH2_CLIENT_ID']
GOOGLE_OAUTH2_CLIENT_SECRET = os.environ['GOOGLE_OAUTH2_CLIENT_SECRET']
GOOGLE_WHITE_LISTED_DOMAINS = ['incuna.com']
SOCIAL_AUTH_USER_MODEL = 'auth.User'
view raw settings.py hosted with ❤ by GitHub

Here I whitelist our Google Apps domain to only allow authentication by users from work email addresses and tell Social Auth to use the auth.User model when creating new users which it will do by default (I believe you can turn this off with another setting). This lets met forget about registration completely which is perfect for internal applications.

Make sure you’ve set GOOGLE_OAUTH2_CLIENT_ID and GOOGLE_OAUTH2_CLIENT_SECRET in your environment when you do runserver or you’ll get the crappy settings error message Unknown command: 'runserver'. You can avoid this using .get() instead of square braces notation when getting the Google credentials however the error may be more archaic. I’m currently favouring square braces notation and getting used to fixing my environment!

Create Some Basic Views Link to heading

from django.core.urlresolvers import reverse
from django.contrib import messages
from django.http import HttpResponse, HttpResponseRedirect
from django.views.generic.base import View
from social_auth.backends.exceptions import AuthFailed
from social_auth.views import complete
class AuthComplete(View):
def get(self, request, *args, **kwargs):
backend = kwargs.pop('backend')
try:
return complete(request, backend, *args, **kwargs)
except AuthFailed:
messages.error(request, "Your Google Apps domain isn't authorized for this app")
return HttpResponseRedirect(reverse('login'))
class LoginError(View):
def get(self, request, *args, **kwargs):
return HttpResponse(status=401)
view raw views.py hosted with ❤ by GitHub

Social Auth requires you add a view for when login fails. So far this hasn’t been an issue for me so I’ve done the pure basics here with LoginError.

The second view was to cope with the whitelisting of domains which, pleasingly, raises an AuthFailed exception when you try to authenticate with a domain not in the whitelist.

Now all we need is to plumb this in with some URLs:

from django.conf.urls import *
from django.contrib import admin
from .views import AuthComplete, LoginError
admin.autodiscover()
urlpatterns = patterns('',
# some other urls
url(r'^admin/', include(admin.site.urls)),
url(r'^complete/(?P<backend>[^/]+)/$', AuthComplete.as_view()),
url(r'^login-error/$', LoginError.as_view()),
url(r'', include('social_auth.urls')),
)
view raw urls.py hosted with ❤ by GitHub

Profit Link to heading

I usually put login_required on the whole site with a little piece of middleware (like this or this) but otherwise that’s it!