Google OAuth Integration in Django Without Utilizing django-allauth
- General
Google OAuth Integration in Django Without Utilizing django-allauth
Google OAuth integration in Django project can sometimes seem daunting, especially with the myriad of third-party solutions available. This guide will provide a methodical walkthrough to integrate Google OAuth, devoid of the django-allauth
solution.
Source Code: Link
Directory Structure:
.
├── README.md
├── django_gauth ──────────── Project Root
│ ├── __init__.py
│ ├── asgi.py
│ ├── backends.py
│ ├── constants.py
│ ├── settings.py
│ ├── urls.py
│ ├── views.py
│ └── wsgi.py
├── manage.py
├── requirements.txt
├── templates
│ └── admin
│ └── login.html
└── venv
STEP 1. Setting Up Google Credentials
- Begin by visiting the Google Cloud Console and create a new project relevant to your application.
- Navigate to API and Services > Credentials from the menu.
- Click on “Create Credentials” and select “OAuth Client ID”.
- Designate the application type as “Web application” and name it appropriately in relation to your project.
- Include your backend URI in “Authorized JavaScript Origins”. If you’re working locally, it’s typically
http://localhost:8000
. - Ensure you add an “Authorized Redirect URI”, set to:
http://localhost:8000/admin/google/callback.
- Remember to note down your Client ID and Client Secret for later use for Google OAuth Integration in Django.
STEP 2. Including Essential Libraries
Ensure your project is equipped with the following:
requests
: Essential for interacting with Google’s OAuth endpoints.
pip install requests
STEP 3. Google OAuth: A Technical Overview
For in-depth understanding, consider referencing Google’s OAuth Documentation.
Google OAuth consists of several key endpoints and steps for user authentication and authorization:
1. Authorization URI:
-
- Endpoint:
https://accounts.google.com/o/oauth2/v2/auth
- Parameters:
scope
: Specifies the requested access scope, e.g., https://www.googleapis.com/auth/userinfo.email.access_type
: Indicates whether offline access is requested.include_granted_scopes
: Optionally includes granted scopes in the response.response_type
: Specifies the response type as code.state
: Optional state parameter for maintaining session state.redirect_uri
: The URL where the user will be redirected after authorization.client_id
: The unique identifier for your application
- Endpoint:
Example URL:
xxxxxxxxxx
https://accounts.google.com/o/oauth2/v2/auth?scope=https%3A//www.googleapis.com/auth/drive.metadata.readonly&access_type=offline&include_granted_scopes=true&response_type=code&state=state_parameter_passthrough_value&redirect_uri=https%3A//oauth2.example.com/code&client_id=client_id
2. Redirect URI:
Upon success, the user is redirected to a specified redirect URI with an authorization code.
-
- Success Response:
Redirect URL:https:///admin/google/callback?code=4/P7q7W91a-oMsCeLvIaQm6bTrgtp7
- Failure Response:
Redirect URL:https:///admin/google/callback?error=access_denied
- Success Response:
3. Exchange Auth Code for Access Token:
To exchange the authorization code for an access token, make a POST request to https://oauth2.googleapis.com/token
with the following JSON parameters:
{
"access_token": "<access_token>",
"expires_in": 3920,
"token_type": "Bearer",
"scope": "https://www.googleapis.com/auth/userinfo.email",
"refresh_token": "<refresh_token>"
}
4. Retrieve Access Code:
This process allows your application to obtain an access token, which can be used to authenticate API requests on behalf of the user for the specified scope. The access token has a limited lifespan, but it can be refreshed using the provided refresh token to maintain long-term access to Google services.
5. Use Access Token to Retrieve User Email
- To Get User details use the access code to api
https://www.googleapis.com/oauth2/v2/userinfo?access_token=
Overall, this OAuth flow facilitates secure and authorized access to Google services for your application while preserving user privacy and control.
STEP 4. Adding following in constants.py
./django_gauth/constants.py
GOOGLE_CLIENT_ID = <YOUR GOOGLE CLIENT ID>
GOOGLE_CLIENT_SECRET = <YOUR GOOGLE CLIENT SECRET>
BASE_URI = 'http://localhost:8000'
GOOGLE_REDIRECT_URI = f"{BASE_URI}/admin/google/callback"
GOOGLE_SCOPES = "https://www.googleapis.com/auth/userinfo.email"
GOOGLE_LOGIN_REDIRECT_URI = (f"https://accounts.google.com/o/oauth2/v2/auth?"
f"response_type={'code'}"
f"&scope={GOOGLE_SCOPES}"
f"&access_type={'offline'}"
f"&include_grant_scopes={'true'}"
f"&state={<SESSION STATE>}"
f"&client_id={GOOGLE_CLIENT_ID}"
f"&redirect_uri={GOOGLE_REDIRECT_URI}")
STEP 5. Crafting a Custom Google Authentication Backend for Google OAuth in Django
This step is pivotal, as it involves instructing Django on the method to validate a user authenticated via Google:
In this step, we will be creating a custom backend that consists of an authenticate method which returns a user instance on successful authentication and None on failure. This user instance will later be used for login.
./django_gauth/backends.py
from django.contrib.auth.backends import BaseBackend
from django.contrib.auth import get_user_model
import requests
from django_gauth import constants
UserModel = get_user_model()
class GoogleAuthBackend(BaseBackend):
"""Custom Backend Server for Google auth"""
def _get_access_token(self, code):
"""
Return access_toke from code
:param code: google Code from callback
:return: User Instance
"""
response = requests.post('https://oauth2.googleapis.com/token', data={
"code": code,
"client_id": constants.GOOGLE_CLIENT_ID,
'client_secret': constants.GOOGLE_CLIENT_SECRET,
"redirect_uri": constants.GOOGLE_REDIRECT_URI,
"grant_type": "authorization_code"
})
return response.json().get('access_token')
def get_user(self, pk):
"""Returns a user instance """
try:
return UserModel.objects.get(pk=pk)
except UserModel.DoesNotExist:
return None
def authenticate(self, request, code=None, **kwargs):
"""
Authentication function for Custom google token verification
parms:
code - Google code received form view
"""
if code:
access_token = self._get_access_token(code)
if access_token:
google_user_details = requests.get( f'https://www.googleapis.com/oauth2/v2/userinfo?access_token={access_token}')
email = google_user_details.json().get('email')
try:
user = UserModel.objects.get(username=email)
return user
except UserModel.DoesNotExist:
return None
STEP 6. Modifying settings.py
To confirm Django acknowledges our custom backend, it must be integrated into the settings.
Update settings.py
./django_gauth/settings.py
AUTHENTICATION_BACKENDS = [
'django_gauth.backends.GoogleAuthBackend',
'django.contrib.auth.backends.ModelBackend'
]
STEP 7. Implementing Views
These are the view functions that handles user request:
We will be defining two views:
google_login
will redirect user to google consent URIgoogle_callback
will listen for the response from the google server
/django_gauth/views.py
from django.http.response import HttpResponseRedirect
from django.shortcuts import redirect
from . import constants
from django.contrib.auth import authenticate, login
from .backends import GoogleAuthBackend
from django.contrib import messages
def google_login(request):
return HttpResponseRedirect(constants.GOOGLE_LOGIN_REDIRECT_URI)
def google_callback(request):
if 'error' in request.GET:
return redirect('admin:index')
if 'code' in request.GET:
user = authenticate(request, code=request.GET.get('code'), backend=GoogleAuthBackend)
if user:
login(request, user=user)
else:
messages.error(request, "You are not Authorized to Login ")
return redirect('admin:index')
STEP 8. Associating Views with URLs
Each view must correspond to a specific URL, ensuring user navigation is coherent, So we will update urls.py
to redirect request to views/controllers
./django_gauth/urls.py
import django_gauth.views
urlpatterns = [
path('admin/google/login', django_gauth.views.google_login, name='login'),
path('admin/google/callback', django_gauth.views.google_callback, name='login_callback'),
path('admin/', admin.site.urls),
]
STEP 9. Overriding default login Template
The Default login template has to be overridden to add a custom Sign in with google Button:
Update./templates/admin/login.html
./templates/admin/login.html
<div class="container">
<a href="/admin/google/login" id="google-signin-button">
<img decoding="async" src="https://developers.google.com/identity/images/g-logo.png" alt="Google Icon">
<span class="text">Sign in with Google</span>
</a>
</div>
Styles to format the Sign in with google button:
python3 manage.py makemigrations
python3 manage.py migrate
python3 manage.py runserver
STEP 10. Test the application
Execute following commands to test your application.
/templates/admin/login.html
<style>
.container {
display: flex;
justify-content: center;
align-items: center;
padding:1rem;
}
#google-signin-button {
display: flex;
align-items: center;
background-color: #fff;
border: 1px solid #ddd;
border-radius: 4px;
padding: 10px 20px;
text-decoration: none;
color: #333;
cursor: pointer;
transition: background-color 0.3s ease;
}
#google-signin-button:hover {
background-color: #f0f0f0;
}
#google-signin-button > img{
margin-right: 10px;
height: 1.5rem;
}
.text {
font-weight: bold;
}
</style>
Visit http://localhost:8000/admin
to test your application
NOTE: Access to the Admin Panel is restricted to a few users who have to be added by the super admin in order to use the “sign in with Google” feature. As a result, all users must be created in the admin panel by the super admin.
In conclusion, you have now adeptly integrated Google OAuth into your Django project, independent of django-allauth
. It’s advised to run and test your project to ensure seamless functionality. Best of luck with your endeavors.
Related content
Auriga: Leveling Up for Enterprise Growth!
Auriga’s journey began in 2010 crafting products for India’s