React Social Authentication using Djoser and Django
Djoser is a powerful authentication library. It offers registration, account activation, login, password reset, and logout features. <!--more-->
Introduction
Djoser offers social authentication, which will be the main focus of this article. We will create an application that handles Google social authentication with a Django backend and a React frontend.
Prerequisites
The reader should have these prerequisites to follow along with this article:
- Python installed on your machine.
- Installed
django
,djangorestframework
,djangorestframework-simplejwt
,djoser
,social-auth-app-django
,django-cors-headers
. You can install them usingpip
. - Have
npm
installed for our frontend part.
Getting started
Start by creating a new directory, then run django-admin startapp backend
under this directory to create a new Django application called backend
.
Djoser
works with a custom user model in a Django application, so let us create a new app inside our project to contain the custom user model. Navigate the backend folder and run python3 manage.py startapp users
.
We need to extend the default User model that comes with Django. In the users
folder, edit the models.py
as shown:
users/models.py
from django.contrib.auth.models import BaseUserManager
from django.contrib.auth.models import AbstractBaseUser, PermissionsMixin
from django.db import models
# Create your models here.
class UserManager(BaseUserManager):
def create_user(self, email, password=None, **kwargs):
if not email:
raise ValueError("Users must have an email address")
email = self.normalize_email(email)
user = self.model(email=email, **kwargs)
user.set_password(password)
user.save()
return user
def create_superuser(self, email, password=None, **kwargs):
kwargs.setdefault('is_active', True)
kwargs.setdefault('is_staff', True)
kwargs.setdefault('is_superuser', True)
if kwargs.get('is_active') is not True:
raise ValueError('Superuser must be active')
if kwargs.get('is_staff') is not True:
raise ValueError('Superuser must be staff')
if kwargs.get('is_superuser') is not True:
raise ValueError('Superuser must have is_superuser=True')
return self.create_user(email, password, **kwargs)
class User(AbstractBaseUser, PermissionsMixin):
email = models.EmailField(max_length=255, unique=True)
first_name = models.CharField(max_length=255)
last_name = models.CharField(max_length=255)
is_active = models.BooleanField(default=False)
is_staff = models.BooleanField(default=False)
objects = UserManager()
USERNAME_FIELD = 'email'
REQUIRED_FIELDS = ['first_name', 'last_name']
def get_full_name(self):
return f"{self.first_name}{self.last_name}"
def get_short_name(self):
return self.first_name
def __str__(self):
return self.email
We have created a new User model and specified the username field as an email address. We have also created a UserManager
model that handles the creation of users and superusers.
Next, we have to edit our settings.py
to set up djoser
and Google authentication.
Add the following to the backend/settings.py
file:
from datetime import timedelta
AUTH_USER_MODEL = 'users.User'
# Application definition
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'users',
'rest_framework',
'djoser',
'corsheaders',
'social_django',
'rest_framework_simplejwt',
'rest_framework_simplejwt.token_blacklist'
]
MIDDLEWARE = [
'corsheaders.middleware.CorsMiddleware',
'social_django.middleware.SocialAuthExceptionMiddleware',
]
CORS_ALLOWED_ORIGINS = [
"http://localhost:3000",
"http://127.0.0.1:3000",
]
CORS_ORIGIN_WHITELIST = [
"http://localhost:3000",
"http://127.0.0.1:3000",
]
CORS_ALLOW_CREDENTIALS = True
ROOT_URLCONF = 'auth_system.urls'
TEMPLATES = [
{
'BACKEND': 'django.template.backends.django.DjangoTemplates',
'DIRS': [],
'APP_DIRS': True,
'OPTIONS': {
'context_processors': [
'django.template.context_processors.debug',
'django.template.context_processors.request',
'django.contrib.auth.context_processors.auth',
'django.contrib.messages.context_processors.messages',
'social_django.context_processors.backends',
'social_django.context_processors.login_redirect'
],
},
},
]
REST_FRAMEWORK = {
'DEFAULT_PERMISSION_CLASSES': [
'rest_framework.permissions.IsAuthenticated',
],
'DEFAULT_AUTHENTICATION_CLASSES': (
'rest_framework_simplejwt.authentication.JWTAuthentication',
),
}
SIMPLE_JWT = {
'AUTH_HEADER_TYPES': ('JWT',),
'ACCESS_TOKEN_LIFETIME': timedelta(minutes=5),
'REFRESH_TOKEN_LIFETIME': timedelta(days=1),
'AUTH_TOKEN_CLASSES': ('rest_framework_simplejwt.tokens.AccessToken',),
}
DJOSER = {
'LOGIN_FIELD': 'email',
'SOCIAL_AUTH_TOKEN_STRATEGY': 'djoser.social.token.jwt.TokenStrategy',
'SOCIAL_AUTH_ALLOWED_REDIRECT_URIS': ['http://127.0.0.1:3000', 'http://127.0.0.1:3000/home','http://127.0.0.1:3000/login'],
'SERIALIZERS': {},
}
AUTHENTICATION_BACKENDS = (
'social_core.backends.google.GoogleOAuth2',
'django.contrib.auth.backends.ModelBackend'
)
SOCIAL_AUTH_GOOGLE_OAUTH2_KEY = 'your_client_id_key'
SOCIAL_AUTH_GOOGLE_OAUTH2_SECRET = 'your_secret_key'
SOCIAL_AUTH_GOOGLE_OAUTH2_SCOPE = [
'https://www.googleapis.com/auth/userinfo.email',
'https://www.googleapis.com/auth/userinfo.profile',
'openid'
]
SOCIAL_AUTH_GOOGLE_OAUTH2_EXTRA_DATA = ['first_name', 'last_name']
We have listed all the apps required by our application in INSTALLED_APPS
. Next, we specified the User
model needed for the authentication and set up the middleware for CORS
and django_social
.
Then we specified the allowed site origins to access our application. This case will be localhost:3000
because we will work with React later.
Then, we made the settings for Django REST Framework and Django REST simple-jwt. Notice the allowed redirect URLs part. These URLs should be similar to those we will set up for the application in the Google cloud console.
We will implement Google for the authentication, so we also need the ModelBackend
to log into the admin panel. Then, we set up settings for SOCIAL_AUTH_GOOGLE_OAUTH2_KEY
, SOCIAL_AUTH_GOOGLE_OAUTH2_SECRET
.
Now we need to configure URLs to make the backend application work.
Edit backend/urls.py
file as follows:
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
path('admin/', admin.site.urls),
path('auth/', include('djoser.urls')),
path('auth/', include('djoser.urls.jwt')),
path('auth/', include('djoser.social.urls')),#Needed for social authentication
]
Run python3 manage.py makemigrations
and python3 manage.py migrate
to finish the backend setup.
Creating Google OAuth credentials
Visit Google Cloud Plaform and create a new project as follows:
With the project selected, click Credentials
and then click OAuth Client ID
after navigating to create credentials.
You may need to set up OAuth consent, so select web application then enter http://127.0.0.1:3000
for authorized JavaScript origins.
For the authorized redirect, URIs have these URLs: http://127.0.0.1:3000
, http://127.0.0.1:3000/home
, and http://127.0.0.1:3000/login
After you click create, you will get your client id
and client secret
, which you will use in your settings.py
as follows:
SOCIAL_AUTH_GOOGLE_OAUTH2_KEY = 'your_client_id_key'
SOCIAL_AUTH_GOOGLE_OAUTH2_SECRET = 'your_secret_key'
Creating the frontend application
In a new terminal session, run npx create-react-app frontend
to create a new React application.
Navigate into frontend and run the following commands npm install axios redux react-redux redux-devtools-extension-redux-thunk styled-components
.
These commands install all the dependencies we need when building this application.
Your project structure should look like:
├── node_modules
├── package.json
├── package-lock.json
├── public
│ ├── favicon.ico
│ ├── images
│ │ ├── google.svg
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
├── README.md
└── src
├── App.css
├── App.js
├── App.test.js
├── index.css
├── index.js
├── logo.svg
│ ├── auth.js
│ └── index.js
├── reportWebVitals.js
└── setupTests.js
Creating the components
Under the src
folder, create a new directory called components
to hold all needed components.
In that folder, create the following components:
1. Components/Home.js
import React from 'react'
import {Link} from "react-router-dom";
const Home = () => {
return (
<div className="container">
<div className="card"style={{ width: 700}}>
<div className="card-body">
<h5 className="card-title">Welcome to The Auth with React & Djoser</h5>
<p className="card-text">Thank you for using this authentication system.</p>
</div>
</div>
</div>
)
}
export default Home
2. Components/Layout.js
import React, from 'react'
const Layout = (props) => {
return (
<div className="container">
<Navbar/>
{props.children}
</div>
)
}
export default Layout
3. Components/Login.js
import React from 'react'
import {Link, Navigate} from 'react-router-dom'
import styled from "styled-components";
const Login = () => {
return (
<div className="container mt-4">
<h1>Sign In</h1>
<p>Log into your account now.</p>
<Google className="btn btn-secondary" >
<img src="/images/google.svg" alt=""/>
Continue with Google.
</Google>
<p className="mt-3">Don't have an account? <Link to="/signup">Register</Link></p>
</div>
)
}
const Google = styled.button`
display: flex;
justify-content: center;
background-color: #fff;
align-items: center;
height: 46px;
width: 30%;
border-radius: 28px;
box-shadow: inset 0 0 0 1px rgb(0, 0, 0, 60%), inset 0 0 0 2px rgb(0, 0, 0, 0%), inset 0 0 0 1px rgb(0, 0, 0, 0);
vertical-align: middle;
z-index: 0;
transition-duration: 167ms;
font-size: 20px;
color: rgbe(0, 0, 0, 0.6);
&:hover {
background-color: rgba(207, 207, 207, 0.25);
color: rgba(0, 0, 0, 0.75);
}
`
export default Login
4. Components/Navbar.js
import React from 'react'
import {Link} from "react-router-dom";
const Navbar = () => {
return(
<nav className="navbar navbar-expand-lg navbar-light bg-light">
<div className="container-fluid">
<a className="navbar-brand" href="#">The Auth</a>
<button className="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav"
aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<span className="navbar-toggler-icon"></span>
</button>
<div className="collapse navbar-collapse" id="navbarNav">
<ul className="navbar-nav">
<li className="nav-item">
<Link className="nav-link active" aria-current="page" to="/">Home</Link>
</li>
<li className="nav-item">
<Link className="nav-link active" aria-current="page" to="/login">Home</Link>
</li>
<li className="nav-item">
<Link className="nav-link active" aria-current="page" to="/signup">Home</Link>
</li>
</ul>
</div>
</div>
</nav>
)
}
export Navbar
5. Components/Signup.js
import React from 'react'
import {Link, Navigate} from 'react-router-dom'
import styled from "styled-components";
const Signup = () => {
return (
<div className="container mt-4">
<h1>Sign Up</h1>
<p>Create into your account now.</p>
<Google className="btn btn-secondary">
<img src="/images/google.svg" alt=""/>
Continue with Google.
</Google>
<p className="mt-3">Have an account? <Link to="/login">Sign In</Link></p>
</div>
)
}
const Google = styled.button`
display: flex;
justify-content: center;
background-color: #fff;
align-items: center;
height: 46px;
width: 30%;
border-radius: 28px;
box-shadow: inset 0 0 0 1px rgb(0, 0, 0, 60%), inset 0 0 0 2px rgb(0, 0, 0, 0%), inset 0 0 0 1px rgb(0, 0, 0, 0);
vertical-align: middle;
z-index: 0;
transition-duration: 167ms;
font-size: 20px;
color: rgbe(0, 0, 0, 0.6);
&:hover {
background-color: rgba(207, 207, 207, 0.25);
color: rgba(0, 0, 0, 0.75);
}
`
export default Signup
6. Components/Welcome.js
import React from 'react'
import {Link} from "react-router-dom";
const Welcome = () => {
return (
<div className="container">
<div className="card"style={{ width: 700}}>
<div className="card-body">
<h5 className="card-title">Welcome to The Auth</h5>
<p className="card-text">Click to Login.</p>
<Link to="/login" className="btn btn-primary">Login</Link>
<p className="mt-3">Don't have an account? <Link to="/signup">Register</Link></p>
</div>
</div>
</div>
)
}
export default Welcome
Now we need to configure routing for our components. Edit the App.js
file as follows:
/* import logo from './logo.svg';*/
/* import './App.css'; */
import React from 'react'
import {BrowserRouter, Route, Routes} from "react-router-dom";
import Welcome from "./components/Welcome";
import Signup from "./components/Signup";
import Login from "./components/Login";
import Layout from "./components/Layout";
import store from "./store";
import Home from "./components/Home";
function App() {
return (
<div>
<BrowserRouter>
<Layout>
<Routes>
<Route exact path="/" element={<Welcome />} />
<Route exact path="/home" element={<Home />} />
<Route path="/login" element={<Login />} />
<Route path="/signup" element={<Signup />} />
</Routes>
</Layout>
</BrowserRouter>
</div>
);
}
export default App;
Use the command npm start
to see this welcome page:
You should also be able to navigate to login and signup pages.
Setting up the authentication
The frontend application will be using redux
for our application. Redux is a state container for JavaScript applications.
We will set up the actions reducers and store files for our application but, first, create the following files and folders under src
in the following folder structure.
├── node_modules
├── package.json
├── package-lock.json
├── public
│ ├── favicon.ico
│ ├── images
│ │ ├── google.svg
│ │ ├── haidong.jpg
│ │ ├── jaspergeys.jpg
│ │ └── robson.jpg
│ ├── index.html
│ ├── logo192.png
│ ├── logo512.png
│ ├── manifest.json
│ └── robots.txt
├── README.md
└── src
├── actions
│ ├── auth.js
│ └── types.js
├── App.css
├── App.js
├── App.test.js
├── components
│ ├── Home.js
│ ├── Layout.js
│ ├── Login.js
│ ├── Navbar.js
│ ├── Signup.js
│ └── Welcome.js
├── index.css
├── index.js
├── logo.svg
├── reducers
│ ├── auth.js
│ └── index.js
├── reportWebVitals.js
├── setupTests.js
└── store.js
- actions - This folder handles how different actions are specified.
- reducers - Handles how states are manipulated in case of different actions.
- store.js - A file that we use to define our store.
- .env file - We use this file to define our default API URL.
We begin by defining our API URL in the .env
file.
REACT_APP_API_URL = 'http://127.0.0.1:8000'
To define our actions, we need different action types, so edit the types.js
file as below:
export const GOOGLE_AUTH_SUCCESS = 'GOOGLE_AUTH_SUCCESS'
export const GOOGLE_AUTH_FAIL = 'GOOGLE_AUTH_FAIL'
export const LOGOUT = 'LOGOUT'
Then we define our actions in the auth.js
file as follows:
import {
GOOGLE_AUTH_FAIL,
GOOGLE_AUTH_SUCCESS,
LOGOUT,
} from "./types";
import axios from "axios";
axios.defaults.withCredentials = true;
export const googleAuthenticate = (state, code) => async dispatch =>{
if( state && code && !localStorage.getItem('access')){
const config = {
headers: {
'Content-Type':'application/x-www-form-urlencoded'
}}
const details = {
'state': state,
'code':code
}
const formBody = Object.keys(details).map(key=> encodeURIComponent(key)+'='+encodeURIComponent(details[key])).join('&')
try{
const res = await axios.post(`${process.env.REACT_APP_API_URL}/auth/o/google-oauth2/?${formBody}`, config);
console.log(res.data)
dispatch({
type:GOOGLE_AUTH_SUCCESS,
payload: res.data
})
}catch(err){
dispatch({
type:GOOGLE_AUTH_FAIL
})
console.log(err)
}
}
}
export const logout = () => dispatch => {
dispatch({
type: LOGOUT
})
}
When a user authenticates using Google, a redirect URI containing the state and code will be sent by the application. We will use this code and state to obtain user information such as access and refresh tokens, email, and name.
The googleAuthenticate
function handles the post request with the code and state as data, then returns the user data as a response.
These results are dispatched to the reducer, with the action type being GOOGLE_AUTH_SUCCESS
. However, in case of any errors, then the action type dispatched is GOOGLE_AUTH_FAIL
.
We also have the logout
action that dispatches the action type LOGOUT
. Next, we need to handle the reducers. We have to create an index.js
file in the src/reducers/
with more than two reducers as below:
import { combineReducers } from 'redux';
import auth from './auth';
const rootReducer = combineReducers({
auth
})
export default rootReducer
We have the auth.js
file under the reducers directory to trigger an action according to type.
import {
GOOGLE_AUTH_FAIL,
GOOGLE_AUTH_SUCCESS,
LOGOUT,
} from "../actions/types";
const initialState = {
access: localStorage.getItem('access'),
refresh: localStorage.getItem('refresh'),
isAuthenticated: false,
}
export default function(state=initialState,action){
switch (action.type){
case GOOGLE_AUTH_SUCCESS:
console.log(action.payload)
localStorage.setItem('access',action.payload.access)
return{
...state,
isAuthenticated: true,
access: action.payload.access,
refresh: action.payload.refresh
}
case GOOGLE_AUTH_FAIL:
case LOGOUT:
console.log(action.payload)
localStorage.removeItem('access')
localStorage.removeItem('refresh')
return{
...state,
isAuthenticated: false,
access: null,
refresh: null
}
default:
return state
}
}
We need to set an initial state that sets default values for access and refresh tokens. We also set a default boolean value of false for isAuthenticated
. This value will later be used to handle routing for authenticated users.
We set access and refresh token values in local storage to the one dispatched if the authentication was successful. The isAuthenticated
is true; otherwise, the token values are null, and isAuthenticated
remains false.
If there is no other action, we return the current state and create our store. Edit the src/store.js
to resemble this:
import { createStore, applyMiddleware } from 'redux';
import { composeWithDevTools } from 'redux-devtools-extension';
import thunk from 'redux-thunk';
import rootReducer from './reducers';
const initialState = {};
const middleware = [thunk];
const store = createStore(
rootReducer,
initialState,
composeWithDevTools(applyMiddleware(...middleware))
);
export default store;
All components will read the states from this file. We need to make our store available to our components.
Let us add it to src/App.js
.
// All imports remain the same
function App() {
return (
<div>
<Provider store={store}>
<BrowserRouter>
// All code remain the same
</BrowserRouter>
</Provider>
</div>
);
}
export default App;
The line <Provider store={store}>
makes the states available to all components in our application. By now, we have to connect our components to the store. So let us begin with the src/components/Layout.js
file.
import React, {useEffect} from 'react'
import Navbar from "./Navbar";
import {googleAuthenticate} from "../actions/auth";
import {connect} from "react-redux";
import {useLocation} from "react-router-dom";
import queryString from "query-string";
const Layout = (props) => {
const location = useLocation()
useEffect(() => {
const values = queryString.parse(location.search)
const state = values.state ? values.state : null
const code = values.code ? values.code : null
console.log('State: '+ state)
console.log('Code: '+code)
if (state && code){
props.googleAuthenticate(state, code)
}
}, [location])
return (
<div className="container">
<Navbar/>
{props.children}
</div>
)
}
export default connect(null, { googleAuthenticate})(Layout)
We use the connect
function to connect to our store. Then, we dispatch the googleAuthenticate
action and pass it as props to the Layout
function.
We need to get the URL of the current page and key-value pairs of the URL. If the code and state exist, we call the googleAuthenticate
action that takes code and state as parameters.
When we get redirected, the URL has a state and code, acquired and passed to the googleAuthenticate
function.
Next, we connect our Signup
to the store. Edit src/components/Signup.js
as follows:
import React, {useEffect, useState} from 'react'
import {Link, Navigate} from 'react-router-dom'
import {connect, useDispatch, useSelector} from 'react-redux'
import styled from "styled-components";
import axios from "axios";
const Signup = ({ isAuthenticated}) => {
const signupWithGoogle = async () => {
try {
const res = await axios.get(`${process.env.REACT_APP_API_URL}/auth/o/google-oauth2/?redirect_uri=${process.env.REACT_APP_API_URL}/login`)
window.location.replace(res.data.authorization_url)
} catch (err) {
console.log("Error logging in")
}
}
if (isAuthenticated) {
return <Navigate to='/home'/>
}
return (
<div className="container mt-4">
<h1>Sign Up</h1>
<p>Create into your account now.</p>
<Google className="btn btn-secondary" onClick={signupWithGoogle}>
<img src="/images/google.svg" alt=""/>
Continue with Google.
</Google>
<p className="mt-3">Have an account? <Link to="/login">Sign In</Link></p>
</div>
)
}
const Google = styled.button`
display: flex;
justify-content: center;
background-color: #fff;
align-items: center;
height: 46px;
width: 30%;
border-radius: 28px;
box-shadow: inset 0 0 0 1px rgb(0, 0, 0, 60%), inset 0 0 0 2px rgb(0, 0, 0, 0%), inset 0 0 0 1px rgb(0, 0, 0, 0);
vertical-align: middle;
z-index: 0;
transition-duration: 167ms;
font-size: 20px;
color: rgbe(0, 0, 0, 0.6);
&:hover {
background-color: rgba(207, 207, 207, 0.25);
color: rgba(0, 0, 0, 0.75);
}
`
const mapStateToProps = state => ({
isAuthenticated: state.auth.isAuthenticated,
})
export default connect(mapStateToProps, null)(Signup);
When a user clicks to continue with Google during sign up, the signupWithGoogle
function is called. This function sends a post request to the backend with a specified redirect URI.
This URI must be one of the allowed redirect URIs in the Djoser settings. Additionally, the allowed URIs must be the same as those added in your Google cloud console.
The user gets directed to the authorization page, and the specified redirect URI with the state and code included in the URL.
We also have passed the isAuthenticated
state to redirect our users to the home page without having them authenticate again if they are already authenticated.
Now we do the same for the login page.
import React, {useEffect, useState} from 'react'
import {Link, Navigate} from 'react-router-dom'
import {connect, useDispatch} from 'react-redux'
import {login} from "../actions/auth";
import styled from "styled-components";
import axios from "axios";
const Login = ({ isAuthenticated}) => {
const loginWithGoogle = async () =>{
try{
const res = await axios.get(`${process.env.REACT_APP_API_URL}/auth/o/google-oauth2/?redirect_uri=${process.env.REACT_APP_API_URL}/home`)
window.location.replace(res.data.authorization_url)
}catch(err){
console.log("Error logging in")
}
}
if(isAuthenticated){
return <Navigate to="/home" />
}
return (
<div className="container mt-4">
<h1>Sign In</h1>
<p>Log into your account now.</p>
<Google className="btn btn-secondary" onClick={loginWithGoogle} >
<img src="/images/google.svg" alt=""/>
Continue with Google.
</Google>
<p className="mt-3">Don't have an account? <Link to="/signup">Register</Link></p>
</div>
)
}
const Google = styled.button`
display: flex;
justify-content: center;
background-color: #fff;
align-items: center;
height: 46px;
width: 30%;
border-radius: 28px;
box-shadow: inset 0 0 0 1px rgb(0, 0, 0, 60%), inset 0 0 0 2px rgb(0, 0, 0, 0%), inset 0 0 0 1px rgb(0, 0, 0, 0);
vertical-align: middle;
z-index: 0;
transition-duration: 167ms;
font-size: 20px;
color: rgbe(0, 0, 0, 0.6);
&:hover {
background-color: rgba(207, 207, 207, 0.25);
color: rgba(0, 0, 0, 0.75);
}
`
const mapStateToProps = state => ({
isAuthenticated: state.auth.isAuthenticated,
})
export default connect(mapStateToProps, null)(Login);
We have completed the same procedure as the signup one, only that this time, the redirect URI will be for the home page when a user logs in.
We have also passed the state isAuthenticated
to ensure an authenticated user is redirected to the home page.
Finally, we handle the logout process. Edit src/components/Navbar.js
as follows:
import React from 'react'
import {Link} from "react-router-dom";
import {logout} from "../actions/auth";
import {connect} from "react-redux";
const Navbar = ({isAuthenticated, logout}) => {
return(
<nav className="navbar navbar-expand-lg navbar-light bg-light">
<div className="container-fluid">
<a className="navbar-brand" href="#">The Auth</a>
<button className="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav"
aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<span className="navbar-toggler-icon"></span>
</button>
<div className="collapse navbar-collapse" id="navbarNav">
<ul className="navbar-nav">
<li className="nav-item">
<Link className="nav-link active" aria-current="page" to="/">Home</Link>
</li>
{isAuthenticated ?
<li className="nav-item">
<a className="nav-link" href="/" onClick={logout}>Logout</a>
</li>
:
<div className="navbar-nav">
<li className="nav-item">
<Link className="nav-link" to="/login">Log In</Link>
</li>
<li className="nav-item">
<Link className="nav-link" to="/signup">Sign Up</Link>
</li>
</div>
}
</ul>
</div>
</div>
</nav>
)
}
const mapStateToProps = state => ({
isAuthenticated: state.auth.isAuthenticated
})
export default connect(mapStateToProps, { logout })(Navbar)
Once more, we pass the isAuthenticated
state and logout function as props to the Navbar function. Next, we check if a user is authenticated and, if so, log the user out; otherwise, the user can either sign up or log in.
The logout function is called when a user clicks the Logout link. You should now be able to authenticate using a Google account.
You will be redirected to the home page with the option of logging out.
Notice that the user remains authenticated as long as they have signed in to their Google account. This way, the user does not have to keep entering the email and password every time they want to log in.
I recommend running
npm run build
, moving the build folder to the backend directory, setting up the necessary URLs for viewing and settings for static files so that your project runs on localhost:8000 to avoid authentication errors.
Conclusion
You have now handled authentication for Google using Djoser. You can do the same for other social accounts such as Facebook, Twitter, or others. Please refer to Djoser Documentation for more information.
Happy coding!
Peer Review Contributions by: Jerim Kaura