'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