use-refresh-token.ts 3.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. 'use client'
  2. import { useCallback, useEffect, useRef } from 'react'
  3. import { jwtDecode } from 'jwt-decode'
  4. import dayjs from 'dayjs'
  5. import utc from 'dayjs/plugin/utc'
  6. import { useRouter } from 'next/navigation'
  7. import type { CommonResponse } from '@/models/common'
  8. import { fetchNewToken } from '@/service/common'
  9. import { fetchWithRetry } from '@/utils'
  10. dayjs.extend(utc)
  11. const useRefreshToken = () => {
  12. const router = useRouter()
  13. const timer = useRef<NodeJS.Timeout>()
  14. const advanceTime = useRef<number>(5 * 60 * 1000)
  15. const interval = useRef<number>(55 * 60 * 1000)
  16. const getExpireTime = useCallback((token: string) => {
  17. if (!token)
  18. return 0
  19. const decoded = jwtDecode(token)
  20. return (decoded.exp || 0) * 1000
  21. }, [])
  22. const getCurrentTimeStamp = useCallback(() => {
  23. return dayjs.utc().valueOf()
  24. }, [])
  25. const handleError = useCallback(() => {
  26. localStorage?.removeItem('is_refreshing')
  27. localStorage?.removeItem('console_token')
  28. localStorage?.removeItem('refresh_token')
  29. localStorage?.removeItem('last_refresh_time')
  30. router.replace('/signin')
  31. }, [])
  32. const getNewAccessToken = useCallback(async (currentAccessToken: string, currentRefreshToken: string) => {
  33. if (localStorage?.getItem('is_refreshing') === '1')
  34. return null
  35. const currentTokenExpireTime = getExpireTime(currentAccessToken)
  36. let lastRefreshTime = parseInt(localStorage?.getItem('last_refresh_time') || '0')
  37. lastRefreshTime = isNaN(lastRefreshTime) ? 0 : lastRefreshTime
  38. if (getCurrentTimeStamp() + advanceTime.current > currentTokenExpireTime
  39. && lastRefreshTime + interval.current < getCurrentTimeStamp()) {
  40. localStorage?.setItem('is_refreshing', '1')
  41. const [e, res] = await fetchWithRetry(fetchNewToken({
  42. body: { refresh_token: currentRefreshToken },
  43. }) as Promise<CommonResponse & { data: { access_token: string; refresh_token: string } }>)
  44. if (e) {
  45. handleError()
  46. return e
  47. }
  48. const { access_token, refresh_token } = res.data
  49. localStorage?.setItem('is_refreshing', '0')
  50. localStorage?.setItem('last_refresh_time', getCurrentTimeStamp().toString())
  51. localStorage?.setItem('console_token', access_token)
  52. localStorage?.setItem('refresh_token', refresh_token)
  53. const newTokenExpireTime = getExpireTime(access_token)
  54. timer.current = setTimeout(() => {
  55. const consoleTokenFromLocalStorage = localStorage?.getItem('console_token')
  56. const refreshTokenFromLocalStorage = localStorage?.getItem('refresh_token')
  57. if (consoleTokenFromLocalStorage && refreshTokenFromLocalStorage)
  58. getNewAccessToken(consoleTokenFromLocalStorage, refreshTokenFromLocalStorage)
  59. }, newTokenExpireTime - advanceTime.current - getCurrentTimeStamp())
  60. }
  61. else {
  62. const newTokenExpireTime = getExpireTime(currentAccessToken)
  63. timer.current = setTimeout(() => {
  64. const consoleTokenFromLocalStorage = localStorage?.getItem('console_token')
  65. const refreshTokenFromLocalStorage = localStorage?.getItem('refresh_token')
  66. if (consoleTokenFromLocalStorage && refreshTokenFromLocalStorage)
  67. getNewAccessToken(consoleTokenFromLocalStorage, refreshTokenFromLocalStorage)
  68. }, newTokenExpireTime - advanceTime.current - getCurrentTimeStamp())
  69. }
  70. return null
  71. }, [getExpireTime, getCurrentTimeStamp, handleError])
  72. useEffect(() => {
  73. return () => {
  74. clearTimeout(timer.current)
  75. localStorage?.removeItem('is_refreshing')
  76. localStorage?.removeItem('last_refresh_time')
  77. }
  78. }, [])
  79. return {
  80. getNewAccessToken,
  81. }
  82. }
  83. export default useRefreshToken