mirror of
https://github.com/NoFxAiOS/nofx.git
synced 2026-06-06 05:51:19 +08:00
317 lines
9.1 KiB
TypeScript
317 lines
9.1 KiB
TypeScript
import React, { createContext, useContext, useState, useEffect } from 'react'
|
|
import { getSystemConfig } from '../lib/config'
|
|
import { reset401Flag, httpClient } from '../lib/httpClient'
|
|
|
|
interface User {
|
|
id: string
|
|
email: string
|
|
}
|
|
|
|
interface AuthContextType {
|
|
user: User | null
|
|
token: string | null
|
|
login: (
|
|
email: string,
|
|
password: string
|
|
) => Promise<{
|
|
success: boolean
|
|
message?: string
|
|
}>
|
|
loginAdmin: (password: string) => Promise<{
|
|
success: boolean
|
|
message?: string
|
|
}>
|
|
register: (
|
|
email: string,
|
|
password: string,
|
|
betaCode?: string
|
|
) => Promise<{ success: boolean; message?: string }>
|
|
resetPassword: (
|
|
email: string,
|
|
newPassword: string
|
|
) => Promise<{ success: boolean; message?: string }>
|
|
logout: () => void
|
|
isLoading: boolean
|
|
}
|
|
|
|
const AuthContext = createContext<AuthContextType | undefined>(undefined)
|
|
|
|
export function AuthProvider({ children }: { children: React.ReactNode }) {
|
|
const [user, setUser] = useState<User | null>(null)
|
|
const [token, setToken] = useState<string | null>(null)
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
|
|
useEffect(() => {
|
|
// Reset 401 flag on page load to allow fresh 401 handling
|
|
reset401Flag()
|
|
|
|
// Check if admin mode is active (uses cached system config)
|
|
getSystemConfig()
|
|
.then(() => {
|
|
// No longer simulate login in admin mode; check local storage uniformly
|
|
const savedToken = localStorage.getItem('auth_token')
|
|
const savedUser = localStorage.getItem('auth_user')
|
|
if (savedToken && savedUser) {
|
|
setToken(savedToken)
|
|
setUser(JSON.parse(savedUser))
|
|
}
|
|
|
|
setIsLoading(false)
|
|
})
|
|
.catch((err) => {
|
|
console.error('Failed to fetch system config:', err)
|
|
// On error, continue checking local storage
|
|
const savedToken = localStorage.getItem('auth_token')
|
|
const savedUser = localStorage.getItem('auth_user')
|
|
|
|
if (savedToken && savedUser) {
|
|
setToken(savedToken)
|
|
setUser(JSON.parse(savedUser))
|
|
}
|
|
setIsLoading(false)
|
|
})
|
|
}, [])
|
|
|
|
// Listen for unauthorized events from httpClient (401 responses)
|
|
useEffect(() => {
|
|
const handleUnauthorized = () => {
|
|
console.log('Unauthorized event received - clearing auth state')
|
|
// Clear auth state when 401 is detected
|
|
setUser(null)
|
|
setToken(null)
|
|
// Note: localStorage cleanup is already done in httpClient
|
|
}
|
|
|
|
window.addEventListener('unauthorized', handleUnauthorized)
|
|
|
|
return () => {
|
|
window.removeEventListener('unauthorized', handleUnauthorized)
|
|
}
|
|
}, [])
|
|
|
|
const login = async (email: string, password: string) => {
|
|
try {
|
|
const response = await fetch('/api/login', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify({ email, password }),
|
|
})
|
|
|
|
const data = await response.json()
|
|
|
|
if (response.ok) {
|
|
if (data.token) {
|
|
// Reset 401 flag on successful login
|
|
reset401Flag()
|
|
|
|
const userInfo = { id: data.user_id, email: data.email }
|
|
setToken(data.token)
|
|
setUser(userInfo)
|
|
localStorage.setItem('auth_token', data.token)
|
|
localStorage.setItem('auth_user', JSON.stringify(userInfo))
|
|
|
|
// Check and redirect to returnUrl if exists
|
|
const returnUrl = sessionStorage.getItem('returnUrl')
|
|
if (returnUrl) {
|
|
sessionStorage.removeItem('returnUrl')
|
|
window.history.pushState({}, '', returnUrl)
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
} else {
|
|
// Redirect to config page
|
|
window.history.pushState({}, '', '/traders')
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
}
|
|
|
|
return { success: true, message: data.message }
|
|
}
|
|
|
|
// Unexpected success response
|
|
return { success: false, message: data.message || 'Unexpected login response' }
|
|
} else {
|
|
return {
|
|
success: false,
|
|
message: data.error,
|
|
}
|
|
}
|
|
} catch (error) {
|
|
return { success: false, message: 'Login failed, please try again' }
|
|
}
|
|
}
|
|
|
|
const loginAdmin = async (password: string) => {
|
|
try {
|
|
const response = await fetch('/api/admin-login', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ password }),
|
|
})
|
|
const data = await response.json()
|
|
if (response.ok) {
|
|
// Reset 401 flag on successful login
|
|
reset401Flag()
|
|
|
|
const userInfo = {
|
|
id: data.user_id || 'admin',
|
|
email: data.email || 'admin@localhost',
|
|
}
|
|
setToken(data.token)
|
|
setUser(userInfo)
|
|
localStorage.setItem('auth_token', data.token)
|
|
localStorage.setItem('auth_user', JSON.stringify(userInfo))
|
|
|
|
// Check and redirect to returnUrl if exists
|
|
const returnUrl = sessionStorage.getItem('returnUrl')
|
|
if (returnUrl) {
|
|
sessionStorage.removeItem('returnUrl')
|
|
window.history.pushState({}, '', returnUrl)
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
} else {
|
|
// Redirect to dashboard
|
|
window.history.pushState({}, '', '/dashboard')
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
}
|
|
return { success: true }
|
|
} else {
|
|
return { success: false, message: data.error || 'Login failed' }
|
|
}
|
|
} catch (e) {
|
|
return { success: false, message: 'Login failed, please try again' }
|
|
}
|
|
}
|
|
|
|
const register = async (
|
|
email: string,
|
|
password: string,
|
|
betaCode?: string
|
|
) => {
|
|
const requestBody: {
|
|
email: string
|
|
password: string
|
|
beta_code?: string
|
|
} = { email, password }
|
|
if (betaCode) {
|
|
requestBody.beta_code = betaCode
|
|
}
|
|
|
|
try {
|
|
const result = await httpClient.post<{
|
|
token: string
|
|
user_id: string
|
|
email: string
|
|
message: string
|
|
}>('/api/register', requestBody)
|
|
|
|
if (result.success && result.data) {
|
|
// Reset 401 flag on successful login
|
|
reset401Flag()
|
|
|
|
const userInfo = { id: result.data.user_id, email: result.data.email }
|
|
setToken(result.data.token)
|
|
setUser(userInfo)
|
|
localStorage.setItem('auth_token', result.data.token)
|
|
localStorage.setItem('auth_user', JSON.stringify(userInfo))
|
|
|
|
// Check and redirect to returnUrl if exists
|
|
const returnUrl = sessionStorage.getItem('returnUrl')
|
|
if (returnUrl) {
|
|
sessionStorage.removeItem('returnUrl')
|
|
window.history.pushState({}, '', returnUrl)
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
} else {
|
|
// Redirect to config page
|
|
window.history.pushState({}, '', '/traders')
|
|
window.dispatchEvent(new PopStateEvent('popstate'))
|
|
}
|
|
|
|
return {
|
|
success: true,
|
|
message: result.message || result.data.message,
|
|
}
|
|
}
|
|
|
|
// Only business errors reach here (system/network errors were intercepted)
|
|
return {
|
|
success: false,
|
|
message: result.message || 'Registration failed',
|
|
}
|
|
} catch (error) {
|
|
console.error('Auth register error:', error);
|
|
// Re-throw if it's a critical error, or return structured error
|
|
// Since httpClient throws on 500, we should return a structured error response
|
|
// to let the UI display it gracefully without crashing.
|
|
return {
|
|
success: false,
|
|
message: error instanceof Error ? error.message : 'Detailed server error'
|
|
}
|
|
}
|
|
}
|
|
|
|
const resetPassword = async (email: string, newPassword: string) => {
|
|
try {
|
|
const response = await fetch('/api/reset-password', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify({
|
|
email,
|
|
new_password: newPassword,
|
|
}),
|
|
})
|
|
|
|
const data = await response.json()
|
|
|
|
if (response.ok) {
|
|
return { success: true, message: data.message }
|
|
} else {
|
|
return { success: false, message: data.error }
|
|
}
|
|
} catch (error) {
|
|
return { success: false, message: 'Password reset failed, please try again' }
|
|
}
|
|
}
|
|
|
|
const logout = () => {
|
|
const savedToken = localStorage.getItem('auth_token')
|
|
if (savedToken) {
|
|
fetch('/api/logout', {
|
|
method: 'POST',
|
|
headers: { Authorization: `Bearer ${savedToken}` },
|
|
}).catch(() => {
|
|
/* ignore network errors on logout */
|
|
})
|
|
}
|
|
setUser(null)
|
|
setToken(null)
|
|
localStorage.removeItem('auth_token')
|
|
localStorage.removeItem('auth_user')
|
|
}
|
|
|
|
return (
|
|
<AuthContext.Provider
|
|
value={{
|
|
user,
|
|
token,
|
|
login,
|
|
loginAdmin,
|
|
register,
|
|
resetPassword,
|
|
logout,
|
|
isLoading,
|
|
}}
|
|
>
|
|
{children}
|
|
</AuthContext.Provider>
|
|
)
|
|
}
|
|
|
|
export function useAuth() {
|
|
const context = useContext(AuthContext)
|
|
if (context === undefined) {
|
|
throw new Error('useAuth must be used within an AuthProvider')
|
|
}
|
|
return context
|
|
}
|