Add File
This commit is contained in:
441
frontend/src/utils/request.ts
Normal file
441
frontend/src/utils/request.ts
Normal file
@@ -0,0 +1,441 @@
|
||||
// src/services/request.ts
|
||||
import axios, {
|
||||
AxiosError,
|
||||
type AxiosInstance,
|
||||
type AxiosRequestConfig,
|
||||
type AxiosResponse,
|
||||
type InternalAxiosRequestConfig,
|
||||
type CancelTokenSource,
|
||||
} from 'axios'
|
||||
|
||||
import { useCache } from '@/utils/useCache'
|
||||
import { getLocale } from './utils'
|
||||
import { useAssistantStore } from '@/stores/assistant'
|
||||
// import { i18n } from '@/i18n'
|
||||
// const t = i18n.global.t
|
||||
const assistantStore = useAssistantStore()
|
||||
const { wsCache } = useCache()
|
||||
// Response data structure
|
||||
export interface ApiResponse<T = unknown> {
|
||||
code: number
|
||||
data: T
|
||||
message: string
|
||||
success: boolean
|
||||
[key: string]: any // Allow additional fields
|
||||
}
|
||||
|
||||
// Extended request options
|
||||
export interface RequestOptions {
|
||||
silent?: boolean // Silent mode (no error alerts)
|
||||
rawResponse?: boolean // Return raw Axios response
|
||||
customError?: boolean // Custom error handling
|
||||
retryCount?: number // Number of retry attempts
|
||||
}
|
||||
|
||||
// Merged request configuration
|
||||
export interface FullRequestConfig extends AxiosRequestConfig {
|
||||
requestOptions?: RequestOptions
|
||||
}
|
||||
|
||||
// Custom error type
|
||||
export interface RequestError<T = any> extends Error {
|
||||
config: FullRequestConfig
|
||||
code?: string
|
||||
request?: any
|
||||
response?: AxiosResponse<T>
|
||||
isAxiosError: boolean
|
||||
}
|
||||
|
||||
class HttpService {
|
||||
private instance: AxiosInstance
|
||||
private cancelTokenSource: CancelTokenSource
|
||||
|
||||
constructor(config?: AxiosRequestConfig) {
|
||||
this.cancelTokenSource = axios.CancelToken.source()
|
||||
this.instance = axios.create({
|
||||
baseURL: import.meta.env.VITE_API_BASE_URL,
|
||||
timeout: 100000,
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
...config?.headers,
|
||||
},
|
||||
...config,
|
||||
})
|
||||
|
||||
this.setupInterceptors()
|
||||
}
|
||||
|
||||
/* private cancelCurrentRequest(message: string) {
|
||||
this.cancelTokenSource.cancel(message)
|
||||
this.cancelTokenSource = axios.CancelToken.source()
|
||||
} */
|
||||
|
||||
private setupInterceptors() {
|
||||
// Request interceptor
|
||||
this.instance.interceptors.request.use(
|
||||
(config: InternalAxiosRequestConfig) => {
|
||||
// Add auth token
|
||||
const token = wsCache.get('user.token')
|
||||
if (token && config.headers) {
|
||||
config.headers['X-SQLBOT-TOKEN'] = `Bearer ${token}`
|
||||
}
|
||||
if (assistantStore.getToken) {
|
||||
const prefix = assistantStore.getType === 4 ? 'Embedded ' : 'Assistant '
|
||||
config.headers['X-SQLBOT-ASSISTANT-TOKEN'] = `${prefix}${assistantStore.getToken}`
|
||||
if (config.headers['X-SQLBOT-TOKEN']) config.headers.delete('X-SQLBOT-TOKEN')
|
||||
if (
|
||||
assistantStore.getType &&
|
||||
!!(assistantStore.getType % 2) &&
|
||||
assistantStore.getCertificate
|
||||
) {
|
||||
config.headers['X-SQLBOT-ASSISTANT-CERTIFICATE'] = btoa(
|
||||
encodeURIComponent(assistantStore.getCertificate)
|
||||
)
|
||||
}
|
||||
if (!assistantStore.getType || assistantStore.getType === 2) {
|
||||
config.headers['X-SQLBOT-ASSISTANT-ONLINE'] = assistantStore.getOnline
|
||||
}
|
||||
}
|
||||
const locale = getLocale()
|
||||
if (locale) {
|
||||
/* const mapping = {
|
||||
'zh-CN': 'zh-CN',
|
||||
en: 'en-US',
|
||||
tw: 'zh-TW',
|
||||
} */
|
||||
/* const val = mapping[locale] || locale */
|
||||
config.headers['Accept-Language'] = locale
|
||||
}
|
||||
if (config.url?.includes('/xpack_static/') && config.baseURL) {
|
||||
config.baseURL = config.baseURL.replace('/api/v1', '')
|
||||
// Skip auth for xpack_static requests
|
||||
return config
|
||||
}
|
||||
|
||||
/* try {
|
||||
const request_key = LicenseGenerator.generate()
|
||||
config.headers['X-SQLBOT-KEY'] = request_key
|
||||
} catch (e: any) {
|
||||
if (e?.message?.includes('offline')) {
|
||||
this.cancelCurrentRequest('license-key error detected')
|
||||
showLicenseKeyError()
|
||||
}
|
||||
} */
|
||||
|
||||
// Request logging
|
||||
// console.log(`[Request] ${config.method?.toUpperCase()} ${config.url}`)
|
||||
|
||||
return config
|
||||
},
|
||||
(error) => {
|
||||
return Promise.reject(error)
|
||||
}
|
||||
)
|
||||
// Response interceptor
|
||||
this.instance.interceptors.response.use(
|
||||
(response: AxiosResponse) => {
|
||||
// console.log(`[Response] ${response.config.url}`, response.data)
|
||||
|
||||
// Return raw response if configured
|
||||
if ((response.config as FullRequestConfig).requestOptions?.rawResponse) {
|
||||
return response
|
||||
}
|
||||
|
||||
// Handle business logic
|
||||
/* if (response.data?.success !== true) {
|
||||
return Promise.reject(response.data)
|
||||
} */
|
||||
if (response.data?.code === 0) {
|
||||
return response.data.data
|
||||
} else if (response.data?.code) {
|
||||
return Promise.reject(response.data)
|
||||
}
|
||||
return response.data
|
||||
},
|
||||
async (error: AxiosError) => {
|
||||
const config = error.config as FullRequestConfig & { __retryCount?: number }
|
||||
const requestOptions = config?.requestOptions || {}
|
||||
|
||||
// Retry logic for specific status codes
|
||||
const shouldRetry =
|
||||
error.response?.status === 502 &&
|
||||
(config.__retryCount || 0) < (requestOptions.retryCount || 3)
|
||||
|
||||
if (shouldRetry) {
|
||||
config.__retryCount = (config.__retryCount || 0) + 1
|
||||
|
||||
// Exponential backoff
|
||||
await new Promise((resolve) => setTimeout(resolve, 1000 * (config.__retryCount || 1)))
|
||||
|
||||
return this.instance.request(config)
|
||||
}
|
||||
|
||||
// Unified error handling
|
||||
if (!requestOptions.customError && !requestOptions.silent) {
|
||||
this.handleError(error)
|
||||
}
|
||||
|
||||
return Promise.reject(error)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
private handleError(error: AxiosError) {
|
||||
let errorMessage = 'Request error'
|
||||
|
||||
if (error.response) {
|
||||
switch (error.response.status) {
|
||||
case 400:
|
||||
errorMessage = 'Invalid request parameters'
|
||||
break
|
||||
case 401:
|
||||
errorMessage = error.response?.data
|
||||
? error.response.data.toString()
|
||||
: 'Unauthorized, please login again'
|
||||
// Redirect to login page if needed
|
||||
ElMessage({
|
||||
message: errorMessage,
|
||||
type: 'error',
|
||||
showClose: true,
|
||||
})
|
||||
setTimeout(() => {
|
||||
wsCache.delete('user.token')
|
||||
window.location.reload()
|
||||
}, 2000)
|
||||
return
|
||||
// break
|
||||
case 403:
|
||||
errorMessage = 'Access denied'
|
||||
break
|
||||
case 404:
|
||||
errorMessage = 'Resource not found'
|
||||
break
|
||||
case 500:
|
||||
errorMessage = 'Server error'
|
||||
break
|
||||
default:
|
||||
errorMessage = `Server responded with error: ${error.response.status}`
|
||||
}
|
||||
if (error?.response?.data) {
|
||||
errorMessage = error.response.data.toString()
|
||||
}
|
||||
} else if (error.request) {
|
||||
errorMessage = 'No response from server'
|
||||
} else if (axios.isCancel(error)) {
|
||||
errorMessage = 'Request canceled'
|
||||
return // Skip showing cancel messages
|
||||
} else {
|
||||
errorMessage = error['message'] || 'Unknown error'
|
||||
}
|
||||
|
||||
// Show error using UI library (e.g., Element Plus, Ant Design)
|
||||
console.error(errorMessage)
|
||||
/* if (errorMessage?.includes('Invalid license key salt')) {
|
||||
showLicenseKeyError()
|
||||
} */
|
||||
// ElMessage.error(errorMessage)
|
||||
ElMessage({
|
||||
message: errorMessage,
|
||||
type: 'error',
|
||||
showClose: true,
|
||||
})
|
||||
}
|
||||
|
||||
// Cancel all pending requests
|
||||
public cancelRequests(message?: string) {
|
||||
this.cancelTokenSource.cancel(message)
|
||||
// Create new token source for future requests
|
||||
this.cancelTokenSource = axios.CancelToken.source()
|
||||
}
|
||||
|
||||
// Base request method
|
||||
public request<T = any>(config: FullRequestConfig): Promise<T> {
|
||||
return this.instance.request({
|
||||
cancelToken: this.cancelTokenSource.token,
|
||||
...config,
|
||||
})
|
||||
}
|
||||
|
||||
// GET request
|
||||
public get<T = any>(url: string, config?: FullRequestConfig): Promise<T> {
|
||||
return this.request({ ...config, method: 'GET', url })
|
||||
}
|
||||
|
||||
// POST request
|
||||
public post<T = any>(url: string, data?: any, config?: FullRequestConfig): Promise<T> {
|
||||
return this.request({ ...config, method: 'POST', url, data })
|
||||
}
|
||||
|
||||
public async fetchStream(url: string, data?: any, controller?: AbortController): Promise<any> {
|
||||
const token = wsCache.get('user.token')
|
||||
const heads: any = {
|
||||
'Content-Type': 'application/json',
|
||||
}
|
||||
if (token) {
|
||||
heads['X-SQLBOT-TOKEN'] = `Bearer ${token}`
|
||||
}
|
||||
if (assistantStore.getToken) {
|
||||
const prefix = assistantStore.getType === 4 ? 'Embedded ' : 'Assistant '
|
||||
heads['X-SQLBOT-ASSISTANT-TOKEN'] = `${prefix}${assistantStore.getToken}`
|
||||
if (heads['X-SQLBOT-TOKEN']) delete heads['X-SQLBOT-TOKEN']
|
||||
if (
|
||||
assistantStore.getType &&
|
||||
!!(assistantStore.getType % 2) &&
|
||||
assistantStore.getCertificate
|
||||
) {
|
||||
await assistantStore.refreshCertificate()
|
||||
heads['X-SQLBOT-ASSISTANT-CERTIFICATE'] = btoa(
|
||||
encodeURIComponent(assistantStore.getCertificate)
|
||||
)
|
||||
}
|
||||
if (!assistantStore.getType || assistantStore.getType === 2) {
|
||||
heads['X-SQLBOT-ASSISTANT-ONLINE'] = assistantStore.getOnline
|
||||
}
|
||||
}
|
||||
|
||||
/* try {
|
||||
const request_key = LicenseGenerator.generate()
|
||||
heads['X-SQLBOT-KEY'] = request_key
|
||||
} catch (e: any) {
|
||||
if (e?.message?.includes('offline')) {
|
||||
controller?.abort('license-key error detected')
|
||||
showLicenseKeyError()
|
||||
}
|
||||
} */
|
||||
|
||||
const real_url = import.meta.env.VITE_API_BASE_URL
|
||||
return fetch(real_url + url, {
|
||||
method: 'POST',
|
||||
headers: heads,
|
||||
body: JSON.stringify(data),
|
||||
signal: controller?.signal,
|
||||
})
|
||||
}
|
||||
|
||||
// PUT request
|
||||
public put<T = any>(url: string, data?: any, config?: FullRequestConfig): Promise<T> {
|
||||
return this.request({ ...config, method: 'PUT', url, data })
|
||||
}
|
||||
|
||||
// DELETE request
|
||||
public delete<T = any>(url: string, config?: FullRequestConfig): Promise<T> {
|
||||
return this.request({ ...config, method: 'DELETE', url })
|
||||
}
|
||||
|
||||
// PATCH request
|
||||
public patch<T = any>(url: string, data?: any, config?: FullRequestConfig): Promise<T> {
|
||||
return this.request({ ...config, method: 'PATCH', url, data })
|
||||
}
|
||||
|
||||
// File upload
|
||||
public upload<T = any>(
|
||||
url: string,
|
||||
file: File,
|
||||
fieldName = 'file',
|
||||
config?: FullRequestConfig
|
||||
): Promise<T> {
|
||||
const formData = new FormData()
|
||||
formData.append(fieldName, file)
|
||||
|
||||
return this.post(url, formData, {
|
||||
headers: {
|
||||
'Content-Type': 'multipart/form-data',
|
||||
},
|
||||
...config,
|
||||
})
|
||||
}
|
||||
|
||||
// Download file
|
||||
public download(url: string, config?: FullRequestConfig): Promise<Blob> {
|
||||
return this.request<Blob>({
|
||||
...config,
|
||||
method: 'GET',
|
||||
url,
|
||||
responseType: 'blob',
|
||||
})
|
||||
}
|
||||
|
||||
public loadRemoteScript(url: string, id?: string, cb?: any): Promise<HTMLElement> {
|
||||
if (!url) {
|
||||
return Promise.reject(new Error('URL is required to load remote script'))
|
||||
}
|
||||
if (id && document.getElementById(id)) {
|
||||
return Promise.resolve(document.getElementById(id) as HTMLElement)
|
||||
}
|
||||
if (url.startsWith('/')) {
|
||||
const real_url = import.meta.env.VITE_API_BASE_URL.replace('/api/v1', '')
|
||||
url = real_url + url
|
||||
}
|
||||
return new Promise<HTMLElement>((resolve, reject) => {
|
||||
// 改用传统的script标签加载方式
|
||||
const script = document.createElement('script')
|
||||
script.src = url
|
||||
script.id = id || `remote-script-${Date.now()}`
|
||||
|
||||
script.onload = () => {
|
||||
if (cb) cb()
|
||||
resolve(script)
|
||||
}
|
||||
|
||||
script.onerror = (error) => {
|
||||
console.error(`Failed to load script from ${url}:`, error)
|
||||
reject(new Error(`Failed to load script from ${url}`))
|
||||
}
|
||||
|
||||
document.head.appendChild(script)
|
||||
})
|
||||
}
|
||||
/* public loadRemoteScript(url: string, id?: string, cb?: any): Promise<HTMLElement> {
|
||||
if (!url) {
|
||||
return Promise.reject(new Error('URL is required to load remote script'))
|
||||
}
|
||||
if (id && document.getElementById(id)) {
|
||||
return Promise.resolve(document.getElementById(id) as HTMLElement)
|
||||
}
|
||||
return new Promise<HTMLElement>((resolve, reject) => {
|
||||
this.get(url, {
|
||||
responseType: 'text',
|
||||
headers: {
|
||||
'Content-Type': 'application/javascript',
|
||||
},
|
||||
})
|
||||
.then((response: any) => {
|
||||
const script = document.createElement('script')
|
||||
script.textContent = response
|
||||
script.id = id || `remote-script-${Date.now()}`
|
||||
// Append script to head
|
||||
document.head.appendChild(script)
|
||||
if (cb) {
|
||||
cb()
|
||||
}
|
||||
resolve(script)
|
||||
})
|
||||
.catch((error: any) => {
|
||||
console.error(`Failed to load script from ${url}:`, error)
|
||||
reject(new Error(`Failed to load script from ${url}: ${error.message}`))
|
||||
})
|
||||
})
|
||||
} */
|
||||
}
|
||||
|
||||
// Create singleton instance
|
||||
export const request = new HttpService({
|
||||
baseURL: import.meta.env.VITE_API_BASE_URL,
|
||||
})
|
||||
/*
|
||||
const showLicenseKeyError = (msg?: string) => {
|
||||
ElMessageBox.confirm(t('license.error_tips'), {
|
||||
confirmButtonType: 'primary',
|
||||
tip: msg || t('license.offline_tips'),
|
||||
confirmButtonText: t('common.refresh'),
|
||||
cancelButtonText: t('common.cancel'),
|
||||
customClass: 'confirm-no_icon',
|
||||
autofocus: false,
|
||||
callback: (value: string) => {
|
||||
if (value === 'confirm') {
|
||||
window.location.reload()
|
||||
}
|
||||
},
|
||||
})
|
||||
}
|
||||
*/
|
||||
Reference in New Issue
Block a user