Files
solution-erp/fe-admin/src/lib/api.ts
pqhuy1987 66c1a5c170
All checks were successful
Deploy SOLUTION_ERP / build-deploy (push) Successful in 2m52s
[CLAUDE] Rebrand: 3 domain huypham.vn → solutions.com.vn + migrate script
User request: anh trỏ 3 subdomain mới về VPS IP 103.124.94.38:
  - api.huypham.vn        → api.solutions.com.vn
  - admin.huypham.vn      → admin.solutions.com.vn
  - user.huypham.vn       → eoffice.solutions.com.vn

Verified DNS: cả 3 resolve 103.124.94.38 ✓

Update 17 file repo:
FE (4): fe-admin/.env.production + fe-user/.env.production
        (VITE_API_BASE_URL → https://api.solutions.com.vn)
        fe-admin/src/lib/{api,realtime}.ts + fe-user equivalents (comment)
BE (1): appsettings.Production.json.example — CORS AllowedOrigins
CI/CD (1): .gitea/workflows/deploy.yml — smoke test URL
Scripts (3): setup-iis-sites (DomainApi/Admin/User), setup-ssl (3 host),
             deploy-all (verify curls)
Docs (5): STATUS, HANDOFF, PROJECT-MAP, vps-setup, gotchas
Skill (1): iis-deploy-runbook — 3 site table + description
Email admin@huypham.vn giữ nguyên (Let's Encrypt contact — không phải
domain serve).

Thêm scripts/migrate-domains.ps1 — 1-shot VPS migration:
  1. Pre-flight: resolve DNS 3 domain → verify IP VPS khớp
  2. Add HTTP binding mới cho 3 IIS site (giữ binding cũ làm fallback)
  3. Run win-acme xin 3 cert Let's Encrypt qua HTTP-01 challenge
     (auto add HTTPS binding + http→https redirect)
  4. Verify /health/live + /health/ready + 2 FE endpoint
  5. (Optional -RemoveOld) xóa binding huypham.vn sau verify OK
Rollback: nếu fail, binding cũ vẫn active → site serve qua huypham.vn.

Anh chạy trên VPS:
  cd C:\solution-erp\scripts  ;  .\migrate-domains.ps1
  # Sau 1-2 ngày verify stable:
  .\migrate-domains.ps1 -RemoveOld -SkipCert
2026-04-24 09:43:05 +07:00

103 lines
3.0 KiB
TypeScript

import axios, { AxiosError, type InternalAxiosRequestConfig } from 'axios'
export const TOKEN_KEY = 'solution-erp-admin-token'
export const REFRESH_KEY = 'solution-erp-admin-refresh'
export const USER_KEY = 'solution-erp-admin-user'
// Dev: Vite proxy /api → :5443 (vite.config.ts)
// Prod: VITE_API_BASE_URL = https://api.solutions.com.vn (env.production)
const BASE_URL = (import.meta.env.VITE_API_BASE_URL ?? '') + '/api'
export const api = axios.create({
baseURL: BASE_URL,
timeout: 30000,
})
api.interceptors.request.use(config => {
const token = localStorage.getItem(TOKEN_KEY)
if (token) config.headers.Authorization = `Bearer ${token}`
return config
})
// Refresh token flow: khi 401 → thử refresh → retry request gốc. Queue các request song song
// để chỉ 1 refresh call chạy, các request khác chờ token mới.
type QueueItem = {
resolve: (value: unknown) => void
reject: (reason?: unknown) => void
config: InternalAxiosRequestConfig
}
let isRefreshing = false
let queue: QueueItem[] = []
function processQueue(error: unknown, token: string | null) {
queue.forEach(({ resolve, reject, config }) => {
if (error || !token) {
reject(error)
} else {
config.headers.Authorization = `Bearer ${token}`
resolve(api(config))
}
})
queue = []
}
function redirectLogin() {
localStorage.removeItem(TOKEN_KEY)
localStorage.removeItem(REFRESH_KEY)
localStorage.removeItem(USER_KEY)
if (!window.location.pathname.startsWith('/login')) {
window.location.href = '/login'
}
}
api.interceptors.response.use(
response => response,
async (error: AxiosError) => {
const original = error.config as InternalAxiosRequestConfig & { _retry?: boolean }
if (error.response?.status !== 401 || !original || original._retry) {
return Promise.reject(error)
}
// Login/refresh endpoint 401 → không retry (tránh infinite loop)
if (original.url?.includes('/auth/login') || original.url?.includes('/auth/refresh')) {
redirectLogin()
return Promise.reject(error)
}
original._retry = true
const refreshToken = localStorage.getItem(REFRESH_KEY)
if (!refreshToken) {
redirectLogin()
return Promise.reject(error)
}
if (isRefreshing) {
return new Promise((resolve, reject) => {
queue.push({ resolve, reject, config: original })
})
}
isRefreshing = true
try {
const res = await axios.post<{ accessToken: string; refreshToken: string }>(
'/api/auth/refresh',
{ refreshToken },
)
const newToken = res.data.accessToken
localStorage.setItem(TOKEN_KEY, newToken)
localStorage.setItem(REFRESH_KEY, res.data.refreshToken)
processQueue(null, newToken)
original.headers.Authorization = `Bearer ${newToken}`
return api(original)
} catch (refreshErr) {
processQueue(refreshErr, null)
redirectLogin()
return Promise.reject(refreshErr)
} finally {
isRefreshing = false
}
},
)