|
@@ -0,0 +1,92 @@
|
|
|
+'use client'
|
|
|
+import { useCallback, useEffect, useRef } from 'react'
|
|
|
+import { jwtDecode } from 'jwt-decode'
|
|
|
+import dayjs from 'dayjs'
|
|
|
+import utc from 'dayjs/plugin/utc'
|
|
|
+import { useRouter } from 'next/navigation'
|
|
|
+import type { CommonResponse } from '@/models/common'
|
|
|
+import { fetchNewToken } from '@/service/common'
|
|
|
+import { fetchWithRetry } from '@/utils'
|
|
|
+
|
|
|
+dayjs.extend(utc)
|
|
|
+
|
|
|
+const useRefreshToken = () => {
|
|
|
+ const router = useRouter()
|
|
|
+ const timer = useRef<NodeJS.Timeout>()
|
|
|
+ const advanceTime = useRef<number>(5 * 60 * 1000)
|
|
|
+ const interval = useRef<number>(55 * 60 * 1000)
|
|
|
+
|
|
|
+ const getExpireTime = useCallback((token: string) => {
|
|
|
+ if (!token)
|
|
|
+ return 0
|
|
|
+ const decoded = jwtDecode(token)
|
|
|
+ return (decoded.exp || 0) * 1000
|
|
|
+ }, [])
|
|
|
+
|
|
|
+ const getCurrentTimeStamp = useCallback(() => {
|
|
|
+ return dayjs.utc().valueOf()
|
|
|
+ }, [])
|
|
|
+
|
|
|
+ const handleError = useCallback(() => {
|
|
|
+ localStorage?.removeItem('is_refreshing')
|
|
|
+ localStorage?.removeItem('console_token')
|
|
|
+ localStorage?.removeItem('refresh_token')
|
|
|
+ localStorage?.removeItem('last_refresh_time')
|
|
|
+ router.replace('/signin')
|
|
|
+ }, [])
|
|
|
+
|
|
|
+ const getNewAccessToken = useCallback(async (currentAccessToken: string, currentRefreshToken: string) => {
|
|
|
+ if (localStorage?.getItem('is_refreshing') === '1')
|
|
|
+ return null
|
|
|
+ const currentTokenExpireTime = getExpireTime(currentAccessToken)
|
|
|
+ let lastRefreshTime = parseInt(localStorage?.getItem('last_refresh_time') || '0')
|
|
|
+ lastRefreshTime = isNaN(lastRefreshTime) ? 0 : lastRefreshTime
|
|
|
+ if (getCurrentTimeStamp() + advanceTime.current > currentTokenExpireTime
|
|
|
+ && lastRefreshTime + interval.current < getCurrentTimeStamp()) {
|
|
|
+ localStorage?.setItem('is_refreshing', '1')
|
|
|
+ const [e, res] = await fetchWithRetry(fetchNewToken({
|
|
|
+ body: { refresh_token: currentRefreshToken },
|
|
|
+ }) as Promise<CommonResponse & { data: { access_token: string; refresh_token: string } }>)
|
|
|
+ if (e) {
|
|
|
+ handleError()
|
|
|
+ return e
|
|
|
+ }
|
|
|
+ const { access_token, refresh_token } = res.data
|
|
|
+ localStorage?.setItem('is_refreshing', '0')
|
|
|
+ localStorage?.setItem('last_refresh_time', getCurrentTimeStamp().toString())
|
|
|
+ localStorage?.setItem('console_token', access_token)
|
|
|
+ localStorage?.setItem('refresh_token', refresh_token)
|
|
|
+ const newTokenExpireTime = getExpireTime(access_token)
|
|
|
+ timer.current = setTimeout(() => {
|
|
|
+ const consoleTokenFromLocalStorage = localStorage?.getItem('console_token')
|
|
|
+ const refreshTokenFromLocalStorage = localStorage?.getItem('refresh_token')
|
|
|
+ if (consoleTokenFromLocalStorage && refreshTokenFromLocalStorage)
|
|
|
+ getNewAccessToken(consoleTokenFromLocalStorage, refreshTokenFromLocalStorage)
|
|
|
+ }, newTokenExpireTime - advanceTime.current - getCurrentTimeStamp())
|
|
|
+ }
|
|
|
+ else {
|
|
|
+ const newTokenExpireTime = getExpireTime(currentAccessToken)
|
|
|
+ timer.current = setTimeout(() => {
|
|
|
+ const consoleTokenFromLocalStorage = localStorage?.getItem('console_token')
|
|
|
+ const refreshTokenFromLocalStorage = localStorage?.getItem('refresh_token')
|
|
|
+ if (consoleTokenFromLocalStorage && refreshTokenFromLocalStorage)
|
|
|
+ getNewAccessToken(consoleTokenFromLocalStorage, refreshTokenFromLocalStorage)
|
|
|
+ }, newTokenExpireTime - advanceTime.current - getCurrentTimeStamp())
|
|
|
+ }
|
|
|
+ return null
|
|
|
+ }, [getExpireTime, getCurrentTimeStamp, handleError])
|
|
|
+
|
|
|
+ useEffect(() => {
|
|
|
+ return () => {
|
|
|
+ clearTimeout(timer.current)
|
|
|
+ localStorage?.removeItem('is_refreshing')
|
|
|
+ localStorage?.removeItem('last_refresh_time')
|
|
|
+ }
|
|
|
+ }, [])
|
|
|
+
|
|
|
+ return {
|
|
|
+ getNewAccessToken,
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+export default useRefreshToken
|