All checks were successful
Deploy SOLUTION_ERP / build-deploy (push) Successful in 2m52s
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
103 lines
3.0 KiB
TypeScript
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
|
|
}
|
|
},
|
|
)
|