refactor: implement automatic token refresh and credential support for auth API client

This commit is contained in:
Your Name
2026-04-10 18:34:28 +05:30
parent 867839f896
commit a0eec02ee4
4 changed files with 27 additions and 7 deletions
+1 -1
View File
@@ -66,7 +66,7 @@ class TokenGenerateView(TokenObtainPairView):
class RefreshTokenView(TokenRefreshView): class RefreshTokenView(TokenRefreshView):
permission_classes = (permissions.IsAuthenticated,) permission_classes = (permissions.AllowAny,)
def post(self, request, *args, **kwargs): def post(self, request, *args, **kwargs):
refresh_token = request.COOKIES.get(settings.SIMPLE_JWT["AUTH_COOKIE"]) refresh_token = request.COOKIES.get(settings.SIMPLE_JWT["AUTH_COOKIE"])
+22 -2
View File
@@ -1,10 +1,30 @@
import axios from "axios"; import axios from "axios";
const apiClient = axios.create({ const authApiClient = axios.create({
baseURL: `${import.meta.env.VITE_API_URL}/api/auth/`, baseURL: `${import.meta.env.VITE_API_URL}/api/auth/`,
withCredentials: true,
headers: { headers: {
"Content-Type": "application/json", "Content-Type": "application/json",
}, },
}); });
export default apiClient; authApiClient.interceptors.response.use(
(response) => response,
async (error) => {
if (error.response.status === 401) {
// token expired, refresh it
try {
const response = await authApiClient.post("refresh/");
if (response.status === 200) {
// refresh successful, retry the request
return authApiClient(error.config);
}
} catch (error) {
return Promise.reject(error);
}
}
return Promise.reject(error);
},
);
export default authApiClient;
+2 -2
View File
@@ -1,7 +1,7 @@
import { CheckCircleIcon, XCircleIcon } from "@phosphor-icons/react"; import { CheckCircleIcon, XCircleIcon } from "@phosphor-icons/react";
import { useEffect, useRef, useState } from "react"; import { useEffect, useRef, useState } from "react";
import { useNavigate, useParams } from "react-router-dom"; import { useNavigate, useParams } from "react-router-dom";
import apiClient from "../api/apiClient"; import authApiClient from "../api/apiClient";
import Logo from "../components/Logo"; import Logo from "../components/Logo";
export default function Activate() { export default function Activate() {
@@ -20,7 +20,7 @@ export default function Activate() {
const activateAccount = async () => { const activateAccount = async () => {
try { try {
await apiClient.get(`/activate/${uidb64}/${token}/`); await authApiClient.get(`/activate/${uidb64}/${token}/`);
setStatus("success"); setStatus("success");
} catch (err) { } catch (err) {
console.error("Activation error:", err); console.error("Activation error:", err);
+2 -2
View File
@@ -5,7 +5,7 @@ import { useState } from "react";
import { useForm } from "react-hook-form"; import { useForm } from "react-hook-form";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { z } from "zod"; import { z } from "zod";
import apiClient from "../api/apiClient"; import authApiClient from "../api/apiClient";
import Logo from "../components/Logo"; import Logo from "../components/Logo";
import FormField from "../components/ui/FormField"; import FormField from "../components/ui/FormField";
@@ -41,7 +41,7 @@ export default function Register() {
setIsLoading(true); setIsLoading(true);
setApiError(null); setApiError(null);
try { try {
await apiClient.post("/register/", { await authApiClient.post("/register/", {
full_name: data.full_name, full_name: data.full_name,
email: data.email, email: data.email,
password: data.password, password: data.password,