Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 47 additions & 4 deletions src/utils/api.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
/* eslint-disable no-console */
/* eslint-disable no-unused-vars */
import axios from 'axios';
import { StatusCodes } from 'http-status-codes';
import usePopup from '@components/usePopup';
import { clearCookie, setCookie } from './cookie';

const { API_URL_BASE } = process.env;

Expand Down Expand Up @@ -75,6 +77,7 @@ axiosInstance.interceptors.request.use(
if (token) {
config.headers.Authorization = token;
}
config.withCredentials = true;
return config;
},
error => {
Expand All @@ -87,9 +90,38 @@ axiosInstance.interceptors.request.use(

axiosInstance.interceptors.response.use(
config => config,
error => Promise.resolve(error.response),
error => {
console.error(error.stack);
return Promise.reject(error.response);
},
);

function handleStatus(error) {
const { status } = error;

// 401
if (status === StatusCodes.UNAUTHORIZED) {
// redirect to login page
removeToken();
removeUserID();
removeRefreshToken();
window.location.href = '/login';
}

// TODO: Edit here for other status codes
}

// if access_token expired, refresh the token
function checkAccessToken() {
// get new token from cookie
const match = document.cookie.match(/(^| )access_token=([^;]+)/);
if (!match) return;

const token = decodeURIComponent(match[2]);
setToken(token);
clearCookie('access_token');
}

export async function requestAPI(config) {
try {
if (config.needToken && !getToken()) {
Expand All @@ -102,14 +134,17 @@ export async function requestAPI(config) {
delete config.needToken;
const response = await axiosInstance.request(config);

checkAccessToken();

// for test
console.log(response);

// includes 3xx, 4xx responses
// includes 2xx responses
return response;
} catch (error) {
console.error(error);
return null;
handleStatus(error);

return error;
}
}

Expand All @@ -125,6 +160,14 @@ export function removeToken() {
localStorage.removeItem('token');
}

export function setRefreshToken(refreshToken, options = {}) {
setCookie('refresh_token', refreshToken, options);
}

export function removeRefreshToken() {
clearCookie('refresh_token');
}

export function setUserID(userID) {
localStorage.setItem('userID', userID);
}
Expand Down
12 changes: 12 additions & 0 deletions src/utils/cookie.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
export function setCookie(name, value, options = {}) {
const cookie = `${name}=${value}`;
const optionString = Object.entries(options).map(([key, value]) => (
value === true ? `${key}` : `${key}=${value}`
)).join(';');

document.cookie = `${cookie};${optionString}`;
}

export function clearCookie(name) {
setCookie(name, '', { 'max-age': -1 });
}
3 changes: 2 additions & 1 deletion src/views/login/index.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { Button, Box, makeStyles, Container, Typography, Link } from '@material-
import Loading from '@components/Loading';
import UosInput from '@components/UosInput';
import { semesterState } from '@states/Semester';
import { API_LOGIN, API_GET_SEMESTER, requestAPI, getToken, setToken, removeUserID, setUserID } from '@utils/api';
import { API_LOGIN, API_GET_SEMESTER, requestAPI, getToken, setToken, setRefreshToken, removeUserID, setUserID } from '@utils/api';
import { foregroundColor } from '@utils/styles/Colors';
import useLogoLayoutStyles from '@utils/styles/login/LogoLayout';
import useLoginLabelStyles from '@utils/styles/login/LoginLabel';
Expand Down Expand Up @@ -85,6 +85,7 @@ export function LoginBox() {
}

setToken(response.data.token);
setRefreshToken(response.data.refresh);
setUserID(response.data.userId);
await callSemester();
getSocket();
Expand Down