Compare commits
6 Commits
807857618f
...
83d9a08b97
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
83d9a08b97 | ||
|
|
91226fa976 | ||
|
|
c91e038953 | ||
|
|
bd258e19c2 | ||
|
|
cf6ae5ea45 | ||
|
|
4b41e7f2dd |
6
.gitignore
vendored
6
.gitignore
vendored
@@ -549,3 +549,9 @@ backend/desc.md
|
|||||||
|
|
||||||
# Claude Code
|
# Claude Code
|
||||||
.claude/
|
.claude/
|
||||||
|
|
||||||
|
# Registry credentials
|
||||||
|
.env.registry
|
||||||
|
|
||||||
|
# Nginx setup tutorial case
|
||||||
|
tmp-nginx-setup/
|
||||||
|
|||||||
53
backend/PORT_ALLOCATION.md
Normal file
53
backend/PORT_ALLOCATION.md
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
# 后端端口分配规范
|
||||||
|
|
||||||
|
## 规划原则
|
||||||
|
|
||||||
|
- **起始端口**:`20000` 起,远离系统端口和常见开发端口(3000、5000、8000、8080 等)
|
||||||
|
- **百位分段**:每个服务域独占一个百位段(`20xxx`),单域最多容纳 100 个端口
|
||||||
|
- **子段细分**:每段内部再按功能分层,便于快速定位
|
||||||
|
|
||||||
|
## 全局分配表
|
||||||
|
|
||||||
|
| 端口段 | 用途 | 说明 |
|
||||||
|
|--------|------|------|
|
||||||
|
| `20000-20099` | 基础设施 | Nginx 网关、监控、日志、管理后台等 |
|
||||||
|
| `20100-20199` | 用户服务 | `user-service`:账号/邮箱的登录、注册、用户管理 |
|
||||||
|
| `20200-20999` | 预留扩展 | 未来新增服务域 |
|
||||||
|
|
||||||
|
## 用户服务段(20100-20199)细分
|
||||||
|
|
||||||
|
| 子段 | 用途 | 已分配端口 |
|
||||||
|
|------|------|-----------|
|
||||||
|
| `20100-20109` | 数据层 | `20101` Postgres、`20103` Redis |
|
||||||
|
| `20110-20149` | 认证/登录类 | `20111` 账号登录、`20113` 邮箱登录 |
|
||||||
|
| `20150-20189` | 注册/管理类 | `20112` 账号注册、`20114` 邮箱注册 |
|
||||||
|
| `20190-20199` | 预留/调试 | 预留 |
|
||||||
|
|
||||||
|
### user-service 端口明细
|
||||||
|
|
||||||
|
| 服务名 | 宿主机端口 | 容器端口 | 说明 |
|
||||||
|
|--------|-----------|---------|------|
|
||||||
|
| user-postgres | `20101` | `5432` | PostgreSQL |
|
||||||
|
| user-redis | `20103` | `6379` | Redis 缓存 |
|
||||||
|
| user-login-account | `20111` | `8080` | 账号密码登录 |
|
||||||
|
| user-register-account | `20112` | `8080` | 账号注册 |
|
||||||
|
| user-login-email | `20113` | `8080` | 邮箱密码登录 |
|
||||||
|
| user-register-email | `20114` | `8080` | 邮箱注册 |
|
||||||
|
|
||||||
|
## 使用方式
|
||||||
|
|
||||||
|
1. **新增服务前先查表**:确认目标服务域的百位段是否还有空余子段
|
||||||
|
2. **`.env` 覆盖**:`docker-compose.yml` 中端口使用 `${VAR:-default}` 语法,本地冲突时修改 `.env`,不动 compose 文件
|
||||||
|
3. **及时更新本文档**:分配新端口后,同步修改上表并提交
|
||||||
|
|
||||||
|
## 示例 .env
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# user-service/.env
|
||||||
|
USER_POSTGRES_PORT=20101
|
||||||
|
USER_REDIS_PORT=20103
|
||||||
|
USER_LOGIN_ACCOUNT_PORT=20111
|
||||||
|
USER_REGISTER_ACCOUNT_PORT=20112
|
||||||
|
USER_LOGIN_EMAIL_PORT=20113
|
||||||
|
USER_REGISTER_EMAIL_PORT=20114
|
||||||
|
```
|
||||||
@@ -10,8 +10,9 @@ RUN mkdir -p /var/log/nginx /var/www/certbot
|
|||||||
COPY nginx/nginx.conf /etc/nginx/nginx.conf
|
COPY nginx/nginx.conf /etc/nginx/nginx.conf
|
||||||
COPY nginx/conf.d/ /etc/nginx/conf.d/
|
COPY nginx/conf.d/ /etc/nginx/conf.d/
|
||||||
|
|
||||||
# 创建自签名证书(仅用于开发,生产环境应挂载真实证书)
|
# 创建 SSL 目录并生成自签名证书(仅用于开发,生产环境应挂载真实证书)
|
||||||
RUN apk add --no-cache openssl && \
|
RUN mkdir -p /etc/nginx/ssl && \
|
||||||
|
apk add --no-cache openssl && \
|
||||||
openssl req -x509 -nodes -days 365 -newkey rsa:2048 \
|
openssl req -x509 -nodes -days 365 -newkey rsa:2048 \
|
||||||
-keyout /etc/nginx/ssl/key.pem \
|
-keyout /etc/nginx/ssl/key.pem \
|
||||||
-out /etc/nginx/ssl/cert.pem \
|
-out /etc/nginx/ssl/cert.pem \
|
||||||
|
|||||||
29
backend/gateway/docker-compose.yml
Normal file
29
backend/gateway/docker-compose.yml
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
version: "3.8"
|
||||||
|
|
||||||
|
services:
|
||||||
|
gateway:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
container_name: api-gateway
|
||||||
|
ports:
|
||||||
|
- "80:80"
|
||||||
|
- "443:443"
|
||||||
|
volumes:
|
||||||
|
# 开发环境:挂载配置便于热更新,生产环境应内嵌在镜像中
|
||||||
|
- ./nginx/nginx.conf:/etc/nginx/nginx.conf:ro
|
||||||
|
- ./nginx/conf.d:/etc/nginx/conf.d:ro
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
- frontend_asset-helper-network
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "curl", "-f", "http://localhost/health"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 3s
|
||||||
|
start_period: 5s
|
||||||
|
retries: 3
|
||||||
|
|
||||||
|
networks:
|
||||||
|
frontend_asset-helper-network:
|
||||||
|
external: true
|
||||||
@@ -3,25 +3,48 @@ server {
|
|||||||
listen 80 default_server;
|
listen 80 default_server;
|
||||||
listen [::]:80 default_server;
|
listen [::]:80 default_server;
|
||||||
server_name _;
|
server_name _;
|
||||||
|
|
||||||
return 444;
|
return 444;
|
||||||
}
|
}
|
||||||
|
|
||||||
# HTTP 重定向到 HTTPS
|
# HTTP 重定向到 HTTPS(生产域名)
|
||||||
server {
|
server {
|
||||||
listen 80;
|
listen 80;
|
||||||
listen [::]:80;
|
listen [::]:80;
|
||||||
server_name api.example.com;
|
server_name api.example.com;
|
||||||
|
|
||||||
location /.well-known/acme-challenge/ {
|
location /.well-known/acme-challenge/ {
|
||||||
root /var/www/certbot;
|
root /var/www/certbot;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
return 301 https://$server_name$request_uri;
|
return 301 https://$server_name$request_uri;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# 开发环境 - 直接代理,不重定向到 HTTPS
|
||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
listen [::]:80;
|
||||||
|
server_name localhost api-gateway host.docker.internal;
|
||||||
|
|
||||||
|
# 开发环境直接代理,不强制 HTTPS
|
||||||
|
include /etc/nginx/conf.d/services/*.conf;
|
||||||
|
|
||||||
|
# 健康检查
|
||||||
|
location /health {
|
||||||
|
access_log off;
|
||||||
|
return 200 '{"status":"healthy","timestamp":"$time_iso8601"}\n';
|
||||||
|
add_header Content-Type application/json;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 根路径
|
||||||
|
location / {
|
||||||
|
return 200 '{"status":"ok","service":"api-gateway","timestamp":"$time_iso8601"}\n';
|
||||||
|
add_header Content-Type application/json;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
# API 网关主配置
|
# API 网关主配置
|
||||||
server {
|
server {
|
||||||
listen 443 ssl http2;
|
listen 443 ssl http2;
|
||||||
|
|||||||
@@ -1,36 +1,62 @@
|
|||||||
# 用户服务路由
|
# 用户服务路由
|
||||||
location /api/v1/users {
|
|
||||||
# 限流
|
# 账号登录(严格限流)
|
||||||
|
location /api/v1/auth/login/account {
|
||||||
|
limit_req zone=api_strict burst=5 nodelay;
|
||||||
|
limit_conn addr 3;
|
||||||
|
|
||||||
|
rewrite ^/api/v1/auth/login/account$ /login break;
|
||||||
|
proxy_pass http://user_login_account;
|
||||||
|
proxy_http_version 1.1;
|
||||||
|
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_set_header X-Request-ID $request_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 邮箱登录(严格限流)
|
||||||
|
location /api/v1/auth/login/email {
|
||||||
|
limit_req zone=api_strict burst=5 nodelay;
|
||||||
|
limit_conn addr 3;
|
||||||
|
|
||||||
|
rewrite ^/api/v1/auth/login/email$ /login break;
|
||||||
|
proxy_pass http://user_login_email;
|
||||||
|
proxy_http_version 1.1;
|
||||||
|
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_set_header X-Request-ID $request_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 账号注册(通用限流)
|
||||||
|
location /api/v1/users/register/account {
|
||||||
limit_req zone=general burst=20 nodelay;
|
limit_req zone=general burst=20 nodelay;
|
||||||
limit_conn addr 10;
|
limit_conn addr 10;
|
||||||
|
|
||||||
# 代理设置
|
rewrite ^/api/v1/users/register/account$ /register break;
|
||||||
proxy_pass http://user_service;
|
proxy_pass http://user_register_account;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
|
|
||||||
proxy_set_header Host $host;
|
proxy_set_header Host $host;
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
proxy_set_header X-Forwarded-Proto $scheme;
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
proxy_set_header X-Request-ID $request_id;
|
proxy_set_header X-Request-ID $request_id;
|
||||||
|
|
||||||
# WebSocket 支持(如果需要)
|
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
|
||||||
proxy_set_header Connection "upgrade";
|
|
||||||
|
|
||||||
# 缓存控制
|
|
||||||
proxy_cache_bypass $http_upgrade;
|
|
||||||
proxy_no_cache 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# 认证相关接口(严格限流)
|
# 邮箱注册(通用限流)
|
||||||
location /api/v1/auth {
|
location /api/v1/users/register/email {
|
||||||
limit_req zone=api_strict burst=5 nodelay;
|
limit_req zone=general burst=20 nodelay;
|
||||||
limit_conn addr 3;
|
limit_conn addr 10;
|
||||||
|
|
||||||
proxy_pass http://user_service;
|
rewrite ^/api/v1/users/register/email$ /register break;
|
||||||
|
proxy_pass http://user_register_email;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
|
|
||||||
proxy_set_header Host $host;
|
proxy_set_header Host $host;
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
|||||||
@@ -44,22 +44,42 @@ http {
|
|||||||
# 连接限制
|
# 连接限制
|
||||||
limit_conn_zone $binary_remote_addr zone=addr:10m;
|
limit_conn_zone $binary_remote_addr zone=addr:10m;
|
||||||
|
|
||||||
# 上游服务
|
# 上游服务 —— 通过宿主机端口访问各微服务(开发环境)
|
||||||
upstream user_service {
|
# 生产环境应改为容器名:端口,并确保同网络
|
||||||
|
upstream user_login_account {
|
||||||
least_conn;
|
least_conn;
|
||||||
server user-service:8080 max_fails=3 fail_timeout=30s;
|
server host.docker.internal:20111 max_fails=3 fail_timeout=30s;
|
||||||
keepalive 32;
|
keepalive 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
upstream user_register_account {
|
||||||
|
least_conn;
|
||||||
|
server host.docker.internal:20112 max_fails=3 fail_timeout=30s;
|
||||||
|
keepalive 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
upstream user_login_email {
|
||||||
|
least_conn;
|
||||||
|
server host.docker.internal:20113 max_fails=3 fail_timeout=30s;
|
||||||
|
keepalive 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
upstream user_register_email {
|
||||||
|
least_conn;
|
||||||
|
server host.docker.internal:20114 max_fails=3 fail_timeout=30s;
|
||||||
|
keepalive 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 以下服务尚未实现,临时标记为 down,避免启动时 DNS 解析失败
|
||||||
upstream order_service {
|
upstream order_service {
|
||||||
least_conn;
|
least_conn;
|
||||||
server order-service:8080 max_fails=3 fail_timeout=30s;
|
server 127.0.0.1:9999 down;
|
||||||
keepalive 32;
|
keepalive 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
upstream payment_service {
|
upstream payment_service {
|
||||||
least_conn;
|
least_conn;
|
||||||
server payment-service:8080 max_fails=3 fail_timeout=30s;
|
server 127.0.0.1:9999 down;
|
||||||
keepalive 32;
|
keepalive 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
37
backend/scripts/push-image.sh
Executable file
37
backend/scripts/push-image.sh
Executable file
@@ -0,0 +1,37 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
ENV_FILE="${SCRIPT_DIR}/../.env.registry"
|
||||||
|
REGISTRY="registry.fishestlife.com"
|
||||||
|
|
||||||
|
if [[ ! -f "$ENV_FILE" ]]; then
|
||||||
|
echo "错误:找不到 ${ENV_FILE}"
|
||||||
|
echo "请从模板创建并填入账号密码"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# shellcheck source=/dev/null
|
||||||
|
source "$ENV_FILE"
|
||||||
|
|
||||||
|
if [[ -z "${REGISTRY_USER:-}" || -z "${REGISTRY_PASS:-}" ]]; then
|
||||||
|
echo "错误:REGISTRY_USER 或 REGISTRY_PASS 未设置"
|
||||||
|
echo "请编辑 ${ENV_FILE} 填入凭证"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "登录镜像仓库 ${REGISTRY} ..."
|
||||||
|
echo "$REGISTRY_PASS" | docker login "$REGISTRY" -u "$REGISTRY_USER" --password-stdin
|
||||||
|
|
||||||
|
# 推送参数:镜像名:标签
|
||||||
|
IMAGE="${1:-}"
|
||||||
|
if [[ -z "$IMAGE" ]]; then
|
||||||
|
echo "用法:$0 <镜像名:标签>"
|
||||||
|
echo "示例:$0 user-service:latest"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "推送镜像 ${IMAGE} ..."
|
||||||
|
docker push "${REGISTRY}/${IMAGE}"
|
||||||
|
|
||||||
|
echo "完成:${REGISTRY}/${IMAGE}"
|
||||||
@@ -14,7 +14,7 @@ services:
|
|||||||
- SERVICE_PORT=8080
|
- SERVICE_PORT=8080
|
||||||
- JWT_SECRET=${JWT_SECRET:-dev-secret-key}
|
- JWT_SECRET=${JWT_SECRET:-dev-secret-key}
|
||||||
ports:
|
ports:
|
||||||
- "8001:8080"
|
- "${USER_LOGIN_ACCOUNT_PORT:-20111}:8080"
|
||||||
depends_on:
|
depends_on:
|
||||||
user-db:
|
user-db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
@@ -41,7 +41,7 @@ services:
|
|||||||
- SERVICE_NAME=user-register-account
|
- SERVICE_NAME=user-register-account
|
||||||
- SERVICE_PORT=8080
|
- SERVICE_PORT=8080
|
||||||
ports:
|
ports:
|
||||||
- "8002:8080"
|
- "${USER_REGISTER_ACCOUNT_PORT:-20112}:8080"
|
||||||
depends_on:
|
depends_on:
|
||||||
user-db:
|
user-db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
@@ -69,7 +69,7 @@ services:
|
|||||||
- SERVICE_PORT=8080
|
- SERVICE_PORT=8080
|
||||||
- JWT_SECRET=${JWT_SECRET:-dev-secret-key}
|
- JWT_SECRET=${JWT_SECRET:-dev-secret-key}
|
||||||
ports:
|
ports:
|
||||||
- "8003:8080"
|
- "${USER_LOGIN_EMAIL_PORT:-20113}:8080"
|
||||||
depends_on:
|
depends_on:
|
||||||
user-db:
|
user-db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
@@ -96,7 +96,7 @@ services:
|
|||||||
- SERVICE_NAME=user-register-email
|
- SERVICE_NAME=user-register-email
|
||||||
- SERVICE_PORT=8080
|
- SERVICE_PORT=8080
|
||||||
ports:
|
ports:
|
||||||
- "8004:8080"
|
- "${USER_REGISTER_EMAIL_PORT:-20114}:8080"
|
||||||
depends_on:
|
depends_on:
|
||||||
user-db:
|
user-db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
@@ -122,7 +122,7 @@ services:
|
|||||||
- user-postgres-data:/var/lib/postgresql/data
|
- user-postgres-data:/var/lib/postgresql/data
|
||||||
- ./migrations:/docker-entrypoint-initdb.d:ro
|
- ./migrations:/docker-entrypoint-initdb.d:ro
|
||||||
ports:
|
ports:
|
||||||
- "5432:5432"
|
- "${USER_POSTGRES_PORT:-20101}:5432"
|
||||||
networks:
|
networks:
|
||||||
- user-network
|
- user-network
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
@@ -138,7 +138,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- user-redis-data:/data
|
- user-redis-data:/data
|
||||||
ports:
|
ports:
|
||||||
- "6379:6379"
|
- "${USER_REDIS_PORT:-20103}:6379"
|
||||||
networks:
|
networks:
|
||||||
- user-network
|
- user-network
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
|
|||||||
@@ -28,18 +28,12 @@ struct LoginRequest {
|
|||||||
password: String,
|
password: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
// 统一响应包装
|
// 登录/认证类接口扁平响应(与前端约定对齐)
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
struct ApiResponse<T> {
|
struct LoginResponse {
|
||||||
success: bool,
|
success: bool,
|
||||||
|
token: Option<String>,
|
||||||
message: String,
|
message: String,
|
||||||
data: Option<T>,
|
|
||||||
}
|
|
||||||
|
|
||||||
// 登录业务数据
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct LoginData {
|
|
||||||
token: String,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// JWT Claims
|
// JWT Claims
|
||||||
@@ -98,7 +92,7 @@ async fn main() {
|
|||||||
async fn login_handler(
|
async fn login_handler(
|
||||||
State(state): State<Arc<AppState>>,
|
State(state): State<Arc<AppState>>,
|
||||||
Json(payload): Json<LoginRequest>,
|
Json(payload): Json<LoginRequest>,
|
||||||
) -> (StatusCode, Json<ApiResponse<LoginData>>) {
|
) -> (StatusCode, Json<LoginResponse>) {
|
||||||
info!("Login attempt for user: {}", payload.username);
|
info!("Login attempt for user: {}", payload.username);
|
||||||
|
|
||||||
// 查询用户账号与密码
|
// 查询用户账号与密码
|
||||||
@@ -120,16 +114,16 @@ async fn login_handler(
|
|||||||
match verify(&payload.password, &password_hash) {
|
match verify(&payload.password, &password_hash) {
|
||||||
Ok(true) => {
|
Ok(true) => {
|
||||||
info!("User {} logged in successfully", payload.username);
|
info!("User {} logged in successfully", payload.username);
|
||||||
|
|
||||||
// 生成 JWT
|
// 生成 JWT
|
||||||
let token = generate_token(&user_id.to_string(), &state.jwt_secret);
|
let token = generate_token(&user_id.to_string(), &state.jwt_secret);
|
||||||
|
|
||||||
(
|
(
|
||||||
StatusCode::OK,
|
StatusCode::OK,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: true,
|
success: true,
|
||||||
|
token: Some(token),
|
||||||
message: "Login successful".to_string(),
|
message: "Login successful".to_string(),
|
||||||
data: Some(LoginData { token }),
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -137,10 +131,10 @@ async fn login_handler(
|
|||||||
warn!("Invalid password for user {}", payload.username);
|
warn!("Invalid password for user {}", payload.username);
|
||||||
(
|
(
|
||||||
StatusCode::UNAUTHORIZED,
|
StatusCode::UNAUTHORIZED,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: false,
|
success: false,
|
||||||
|
token: None,
|
||||||
message: "Invalid credentials".to_string(),
|
message: "Invalid credentials".to_string(),
|
||||||
data: None,
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -148,10 +142,10 @@ async fn login_handler(
|
|||||||
warn!("Password verification error: {:?}", e);
|
warn!("Password verification error: {:?}", e);
|
||||||
(
|
(
|
||||||
StatusCode::INTERNAL_SERVER_ERROR,
|
StatusCode::INTERNAL_SERVER_ERROR,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: false,
|
success: false,
|
||||||
|
token: None,
|
||||||
message: "Internal error".to_string(),
|
message: "Internal error".to_string(),
|
||||||
data: None,
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,18 +28,12 @@ struct LoginRequest {
|
|||||||
password: String,
|
password: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
// 统一响应包装
|
// 登录/认证类接口扁平响应(与前端约定对齐)
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
struct ApiResponse<T> {
|
struct LoginResponse {
|
||||||
success: bool,
|
success: bool,
|
||||||
|
token: Option<String>,
|
||||||
message: String,
|
message: String,
|
||||||
data: Option<T>,
|
|
||||||
}
|
|
||||||
|
|
||||||
// 登录业务数据
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct LoginData {
|
|
||||||
token: String,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// JWT Claims
|
// JWT Claims
|
||||||
@@ -98,7 +92,7 @@ async fn main() {
|
|||||||
async fn login_handler(
|
async fn login_handler(
|
||||||
State(state): State<Arc<AppState>>,
|
State(state): State<Arc<AppState>>,
|
||||||
Json(payload): Json<LoginRequest>,
|
Json(payload): Json<LoginRequest>,
|
||||||
) -> (StatusCode, Json<ApiResponse<LoginData>>) {
|
) -> (StatusCode, Json<LoginResponse>) {
|
||||||
info!("Login attempt for email: {}", payload.email);
|
info!("Login attempt for email: {}", payload.email);
|
||||||
|
|
||||||
// 查询用户邮箱与密码
|
// 查询用户邮箱与密码
|
||||||
@@ -120,16 +114,16 @@ async fn login_handler(
|
|||||||
match verify(&payload.password, &password_hash) {
|
match verify(&payload.password, &password_hash) {
|
||||||
Ok(true) => {
|
Ok(true) => {
|
||||||
info!("Email {} logged in successfully", payload.email);
|
info!("Email {} logged in successfully", payload.email);
|
||||||
|
|
||||||
// 生成 JWT
|
// 生成 JWT
|
||||||
let token = generate_token(&user_id.to_string(), &state.jwt_secret);
|
let token = generate_token(&user_id.to_string(), &state.jwt_secret);
|
||||||
|
|
||||||
(
|
(
|
||||||
StatusCode::OK,
|
StatusCode::OK,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: true,
|
success: true,
|
||||||
|
token: Some(token),
|
||||||
message: "Login successful".to_string(),
|
message: "Login successful".to_string(),
|
||||||
data: Some(LoginData { token }),
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -137,10 +131,10 @@ async fn login_handler(
|
|||||||
warn!("Invalid password for email {}", payload.email);
|
warn!("Invalid password for email {}", payload.email);
|
||||||
(
|
(
|
||||||
StatusCode::UNAUTHORIZED,
|
StatusCode::UNAUTHORIZED,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: false,
|
success: false,
|
||||||
|
token: None,
|
||||||
message: "Invalid credentials".to_string(),
|
message: "Invalid credentials".to_string(),
|
||||||
data: None,
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -148,10 +142,10 @@ async fn login_handler(
|
|||||||
warn!("Password verification error: {:?}", e);
|
warn!("Password verification error: {:?}", e);
|
||||||
(
|
(
|
||||||
StatusCode::INTERNAL_SERVER_ERROR,
|
StatusCode::INTERNAL_SERVER_ERROR,
|
||||||
Json(ApiResponse {
|
Json(LoginResponse {
|
||||||
success: false,
|
success: false,
|
||||||
|
token: None,
|
||||||
message: "Internal error".to_string(),
|
message: "Internal error".to_string(),
|
||||||
data: None,
|
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
12
frontend/.dockerignore
Normal file
12
frontend/.dockerignore
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
node_modules
|
||||||
|
npm-debug.log
|
||||||
|
Dockerfile*
|
||||||
|
docker-compose*
|
||||||
|
.dockerignore
|
||||||
|
dist
|
||||||
|
.git
|
||||||
|
.gitignore
|
||||||
|
README.md
|
||||||
|
.vscode
|
||||||
|
.idea
|
||||||
|
*.md
|
||||||
@@ -4,28 +4,53 @@
|
|||||||
|
|
||||||
## 项目状态
|
## 项目状态
|
||||||
|
|
||||||
> **当前状态:尚未初始化代码。** 技术栈待定,目录除本文件外为空。
|
**技术栈已确定,项目已初始化。**
|
||||||
> 在选定技术栈后,请补全本文件中标记为「⚠️ 待补充」的章节。
|
|
||||||
|
| 维度 | 选型 |
|
||||||
|
|------|------|
|
||||||
|
| 框架 | React 18 + TypeScript 5 |
|
||||||
|
| 构建 | Vite 6 |
|
||||||
|
| UI 库 | Ant Design 5 + @ant-design/icons |
|
||||||
|
| 路由 | React Router 6 |
|
||||||
|
| 状态管理 | Zustand(客户端状态) |
|
||||||
|
| HTTP 客户端 | Axios(API 封装在 `src/api/`) |
|
||||||
|
|
||||||
## 定位
|
## 定位
|
||||||
|
|
||||||
`frontend/` 是 asset_helper 的 **Web 前端**,调用后端 API(默认通过 Nginx 网关 `https://api.example.com` 暴露)。
|
`frontend/` 是 asset_helper 的 **Web 管理后台**,调用后端 API(通过 Nginx 网关暴露)。
|
||||||
|
|
||||||
- 与 `backend/` 通过 HTTP/JSON 交互,遵循根目录 [CLAUDE.md](../CLAUDE.md) 中定义的跨端契约
|
- 与 `backend/` 通过 HTTP/JSON 交互,遵循根目录 [CLAUDE.md](../CLAUDE.md) 中定义的跨端契约
|
||||||
- 与 `app/`(移动/桌面端)共享后端,但 UI 实现独立
|
- 与 `app/`(移动/桌面端)共享后端,但 UI 实现独立
|
||||||
|
- 当前为后台管理系统,后续可扩展为面向用户的 Web 端
|
||||||
|
|
||||||
## 技术栈
|
## Docker 开发与部署
|
||||||
|
|
||||||
> ⚠️ 待补充:选定后请填写。建议候选:
|
### 开发环境(热更新,不污染物理机)
|
||||||
>
|
|
||||||
> | 维度 | 候选 |
|
```bash
|
||||||
> |------|------|
|
cd frontend
|
||||||
> | 框架 | React 18 / Vue 3 / Next.js |
|
docker-compose -f docker-compose.dev.yml up --build
|
||||||
> | 语言 | TypeScript(推荐,与 Rust 后端类型对齐更顺) |
|
```
|
||||||
> | 构建 | Vite / Next.js |
|
|
||||||
> | 状态管理 | Zustand / Pinia / React Query |
|
- 访问:`http://localhost:3000`
|
||||||
> | UI 库 | Ant Design / Element Plus / shadcn-ui |
|
- 源码通过 volume 挂载,修改后自动热更新
|
||||||
> | HTTP 客户端 | axios / ky / fetch 封装 |
|
- API 请求通过 Vite proxy 转发到后端网关(默认 `http://host.docker.internal:80`)
|
||||||
|
|
||||||
|
**如需修改后端地址:**
|
||||||
|
```bash
|
||||||
|
VITE_API_BASE_URL=http://your-backend:80 docker-compose -f docker-compose.dev.yml up
|
||||||
|
```
|
||||||
|
|
||||||
|
### 生产构建与部署
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd frontend
|
||||||
|
docker-compose up --build
|
||||||
|
```
|
||||||
|
|
||||||
|
- 访问:`http://localhost:20080`(端口可通过 `ADMIN_WEB_PORT` 环境变量修改)
|
||||||
|
- 多阶段构建:Node 构建 → Nginx 提供静态文件
|
||||||
|
- Nginx 代理 `/api/*` 到后端网关容器
|
||||||
|
|
||||||
## 与后端的协作约定
|
## 与后端的协作约定
|
||||||
|
|
||||||
@@ -35,25 +60,18 @@
|
|||||||
|
|
||||||
```ts
|
```ts
|
||||||
interface ApiRequest<T> {
|
interface ApiRequest<T> {
|
||||||
device: number; // 见下方 device 编码
|
device: number; // 前端固定使用 Device.Web = 3
|
||||||
language: number; // 见下方 language 编码
|
language: number; // 默认 Language.SimplifiedChinese = 1
|
||||||
data: T; // 业务字段
|
data: T; // 业务字段
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
**device 编码(必须与 backend 保持一致):**
|
**device 编码:**
|
||||||
- `1` = iOS
|
- `1` = iOS、`2` = Android、`3` = Web ← **前端使用此值**
|
||||||
- `2` = Android
|
- `4` = iPad、`5` = macOS、`6` = Windows、`7` = Linux
|
||||||
- `3` = Web ← **前端使用此值**
|
|
||||||
- `4` = iPad
|
|
||||||
- `5` = macOS
|
|
||||||
- `6` = Windows
|
|
||||||
- `7` = Linux
|
|
||||||
|
|
||||||
**language 编码:**
|
**language 编码:**
|
||||||
- `1` = 简体中文
|
- `1` = 简体中文(默认)、`2` = 繁体中文、`3` = 英文
|
||||||
- `2` = 繁体中文
|
|
||||||
- `3` = 英文
|
|
||||||
|
|
||||||
### 2. 响应包装
|
### 2. 响应包装
|
||||||
|
|
||||||
@@ -75,7 +93,7 @@ interface LoginResponse {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
JWT 存储建议:HttpOnly Cookie(CSRF 防护)或 localStorage(注意 XSS 风险),后续根据安全策略统一。
|
**JWT 存储策略**:存于 Zustand 内存中(页面刷新丢失,需重新登录)。如需持久化,可改为 localStorage,但需注意 XSS 风险。
|
||||||
|
|
||||||
### 4. 错误响应(HTTP 非 200)
|
### 4. 错误响应(HTTP 非 200)
|
||||||
|
|
||||||
@@ -87,44 +105,88 @@ interface ErrorResponse {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## 目录结构
|
||||||
|
|
||||||
|
```
|
||||||
|
frontend/
|
||||||
|
├── docker/
|
||||||
|
│ ├── Dockerfile # 生产多阶段构建
|
||||||
|
│ ├── Dockerfile.dev # 开发环境(volume 挂载源码)
|
||||||
|
│ └── nginx.conf # 生产 Nginx SPA 配置
|
||||||
|
├── src/
|
||||||
|
│ ├── api/
|
||||||
|
│ │ ├── client.ts # Axios 封装(device/language 注入、JWT、错误处理)
|
||||||
|
│ │ └── auth.ts # 认证相关 API
|
||||||
|
│ ├── components/ # 可复用 UI 组件(待扩展)
|
||||||
|
│ ├── hooks/ # 自定义 Hooks(待扩展)
|
||||||
|
│ ├── layouts/
|
||||||
|
│ │ └── MainLayout.tsx # 后台主布局(侧边栏 + 头部 + 内容区)
|
||||||
|
│ ├── pages/
|
||||||
|
│ │ ├── LoginPage.tsx # 登录页(账号/邮箱切换)
|
||||||
|
│ │ ├── DashboardPage.tsx # 仪表盘首页
|
||||||
|
│ │ └── NotFoundPage.tsx # 404
|
||||||
|
│ ├── router/
|
||||||
|
│ │ └── index.tsx # 路由配置(登录守卫 + 受保护路由)
|
||||||
|
│ ├── stores/
|
||||||
|
│ │ └── auth.ts # Zustand 认证状态(token + login/logout)
|
||||||
|
│ ├── types/
|
||||||
|
│ │ ├── api.ts # 通用 API 类型(device/language 枚举、包装类型)
|
||||||
|
│ │ └── auth.ts # 认证相关类型
|
||||||
|
│ ├── utils/
|
||||||
|
│ │ └── storage.ts # localStorage 封装(带前缀隔离)
|
||||||
|
│ ├── App.tsx # 根组件(ConfigProvider + RouterProvider)
|
||||||
|
│ └── main.tsx # 入口
|
||||||
|
├── docker-compose.dev.yml # 开发编排(热更新)
|
||||||
|
├── docker-compose.yml # 生产编排
|
||||||
|
├── index.html
|
||||||
|
├── package.json
|
||||||
|
├── tsconfig.json / tsconfig.app.json / tsconfig.node.json
|
||||||
|
└── vite.config.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
## API 调用规范
|
||||||
|
|
||||||
|
**必须使用封装函数,禁止直接 fetch/axios:**
|
||||||
|
|
||||||
|
```ts
|
||||||
|
// ✅ 业务接口(自动注入 device/language)
|
||||||
|
import { apiPost, apiGet } from '@/api/client'
|
||||||
|
const data = await apiPost<RequestType, ResponseType>('/api/v1/users/xxx', payload)
|
||||||
|
|
||||||
|
// ✅ 认证接口(扁平格式,不包装)
|
||||||
|
import { loginAccount } from '@/api/auth'
|
||||||
|
const result = await loginAccount({ account: 'xxx', password: 'xxx' })
|
||||||
|
```
|
||||||
|
|
||||||
## 代码风格
|
## 代码风格
|
||||||
|
|
||||||
- 注释使用**中文**(与后端保持一致)
|
- 注释使用**中文**(与后端保持一致)
|
||||||
- TypeScript 类型与后端 Rust 结构体一一对齐,避免 `any`
|
- TypeScript 类型与后端 Rust 结构体一一对齐,禁止 `any`
|
||||||
- API 客户端集中封装在一处(如 `src/api/`),不要在组件中直接 fetch
|
- API 调用集中在 `src/api/`,不在组件中直接写 axios
|
||||||
|
- 路由守卫在 `src/router/index.tsx` 中统一配置
|
||||||
## 目录结构
|
|
||||||
|
|
||||||
> ⚠️ 待补充:技术栈选定后填写。常见骨架:
|
|
||||||
>
|
|
||||||
> ```
|
|
||||||
> frontend/
|
|
||||||
> ├── src/
|
|
||||||
> │ ├── api/ # API 客户端封装(device/language 注入、错误统一处理)
|
|
||||||
> │ ├── components/ # 可复用 UI 组件
|
|
||||||
> │ ├── pages/ # 路由页面
|
|
||||||
> │ ├── stores/ # 状态管理
|
|
||||||
> │ ├── types/ # 与后端对齐的类型定义
|
|
||||||
> │ └── utils/
|
|
||||||
> ├── public/
|
|
||||||
> ├── package.json
|
|
||||||
> └── vite.config.ts (或对应配置)
|
|
||||||
> ```
|
|
||||||
|
|
||||||
## 常用命令
|
## 常用命令
|
||||||
|
|
||||||
> ⚠️ 待补充:技术栈选定后填写(如 `pnpm dev` / `pnpm build` / `pnpm test`)。
|
| 命令 | 说明 |
|
||||||
|
|------|------|
|
||||||
|
| `npm install` | 安装依赖(开发容器内自动执行) |
|
||||||
|
| `npm run dev` | 启动开发服务器(容器内) |
|
||||||
|
| `npm run build` | 生产构建 |
|
||||||
|
| `npm run preview` | 预览生产构建 |
|
||||||
|
|
||||||
## 开发环境
|
## 开发环境
|
||||||
|
|
||||||
- 后端网关默认监听 `localhost:80`/`443`,本地需配置 hosts 或直连 `localhost`
|
- 后端网关默认通过 Docker 网络或 `host.docker.internal` 访问
|
||||||
- 后端开发证书为自签名,浏览器需信任或开发环境关闭 HTTPS
|
- 开发容器内 Vite 监听 `0.0.0.0:5173`,映射到宿主机 `3000`
|
||||||
|
- 如需后端使用 HTTPS 自签名证书,Vite proxy 已配置 `secure: false`
|
||||||
|
|
||||||
## 扩展指南
|
## 扩展指南
|
||||||
|
|
||||||
新增页面/功能时:
|
新增页面/功能时:
|
||||||
1. 先在 `types/` 定义与后端对齐的类型
|
1. 先在 `types/` 定义与后端对齐的类型
|
||||||
2. 在 `api/` 添加调用封装(务必通过统一封装注入 `device`/`language`)
|
2. 在 `api/` 添加调用封装(务必通过统一封装注入 `device`/`language`)
|
||||||
3. 再开发组件/页面
|
3. 在 `pages/` 创建页面组件
|
||||||
|
4. 在 `router/index.tsx` 添加路由
|
||||||
|
5. 如需加入侧边栏菜单,在 `layouts/MainLayout.tsx` 的 `menuItems` 中配置
|
||||||
|
|
||||||
发现重复逻辑时优先抽到 `utils/` 或自定义 Hook(React)/ Composable(Vue)。
|
发现重复逻辑时优先抽到 `utils/` 或自定义 Hook。
|
||||||
|
|||||||
29
frontend/docker-compose.dev.yml
Normal file
29
frontend/docker-compose.dev.yml
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
services:
|
||||||
|
admin-web-dev:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: docker/Dockerfile.dev
|
||||||
|
container_name: asset-helper-admin-dev
|
||||||
|
environment:
|
||||||
|
- NODE_ENV=development
|
||||||
|
- CHOKIDAR_USEPOLLING=true
|
||||||
|
- VITE_API_BASE_URL=${VITE_API_BASE_URL:-http://host.docker.internal:80}
|
||||||
|
ports:
|
||||||
|
- "3000:5173"
|
||||||
|
volumes:
|
||||||
|
# 源码挂载(实现热更新)
|
||||||
|
- ./src:/app/src:ro
|
||||||
|
- ./index.html:/app/index.html:ro
|
||||||
|
- ./vite.config.ts:/app/vite.config.ts:ro
|
||||||
|
- ./tsconfig.json:/app/tsconfig.json:ro
|
||||||
|
- ./tsconfig.app.json:/app/tsconfig.app.json:ro
|
||||||
|
- ./tsconfig.node.json:/app/tsconfig.node.json:ro
|
||||||
|
# 不覆盖 node_modules
|
||||||
|
- /app/node_modules
|
||||||
|
networks:
|
||||||
|
- asset-helper-network
|
||||||
|
restart: unless-stopped
|
||||||
|
|
||||||
|
networks:
|
||||||
|
asset-helper-network:
|
||||||
|
driver: bridge
|
||||||
22
frontend/docker-compose.yml
Normal file
22
frontend/docker-compose.yml
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
version: "3.8"
|
||||||
|
|
||||||
|
services:
|
||||||
|
admin-web:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: docker/Dockerfile
|
||||||
|
container_name: asset-helper-admin
|
||||||
|
ports:
|
||||||
|
- "${ADMIN_WEB_PORT:-20080}:80"
|
||||||
|
networks:
|
||||||
|
- asset-helper-network
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "wget", "--quiet", "--tries=1", "--spider", "http://localhost:80/"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
|
||||||
|
networks:
|
||||||
|
asset-helper-network:
|
||||||
|
driver: bridge
|
||||||
28
frontend/docker/Dockerfile
Normal file
28
frontend/docker/Dockerfile
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# 生产环境 Dockerfile — 多阶段构建
|
||||||
|
# Stage 1: 构建
|
||||||
|
FROM node:20-alpine AS builder
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
COPY package.json package-lock.json* ./
|
||||||
|
RUN npm ci
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
RUN npm run build
|
||||||
|
|
||||||
|
# Stage 2: 运行
|
||||||
|
FROM nginx:1.25-alpine
|
||||||
|
|
||||||
|
# 复制自定义 Nginx 配置
|
||||||
|
COPY docker/nginx.conf /etc/nginx/conf.d/default.conf
|
||||||
|
|
||||||
|
# 复制构建产物
|
||||||
|
COPY --from=builder /app/dist /usr/share/nginx/html
|
||||||
|
|
||||||
|
# 健康检查
|
||||||
|
HEALTHCHECK --interval=30s --timeout=10s --start-period=5s --retries=3 \
|
||||||
|
CMD wget --quiet --tries=1 --spider http://localhost:80/ || exit 1
|
||||||
|
|
||||||
|
EXPOSE 80
|
||||||
|
|
||||||
|
CMD ["nginx", "-g", "daemon off;"]
|
||||||
16
frontend/docker/Dockerfile.dev
Normal file
16
frontend/docker/Dockerfile.dev
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
# 开发环境 Dockerfile
|
||||||
|
# 不复制源码,通过 docker-compose volume 挂载,实现热更新
|
||||||
|
|
||||||
|
FROM node:20-alpine
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# 安装依赖(利用 Docker 缓存层)
|
||||||
|
COPY package.json package-lock.json* ./
|
||||||
|
RUN npm install
|
||||||
|
|
||||||
|
# 暴露 Vite 开发服务器端口
|
||||||
|
EXPOSE 5173
|
||||||
|
|
||||||
|
# 开发模式启动(--host 确保外部可访问)
|
||||||
|
CMD ["npm", "run", "dev", "--", "--host", "0.0.0.0"]
|
||||||
42
frontend/docker/nginx.conf
Normal file
42
frontend/docker/nginx.conf
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
server_name localhost;
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
index index.html;
|
||||||
|
|
||||||
|
# Gzip 压缩
|
||||||
|
gzip on;
|
||||||
|
gzip_vary on;
|
||||||
|
gzip_proxied any;
|
||||||
|
gzip_comp_level 6;
|
||||||
|
gzip_types text/plain text/css text/xml application/json application/javascript application/rss+xml application/atom+xml image/svg+xml;
|
||||||
|
|
||||||
|
# 静态资源缓存
|
||||||
|
location ~* \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2)$ {
|
||||||
|
expires 1y;
|
||||||
|
add_header Cache-Control "public, immutable";
|
||||||
|
}
|
||||||
|
|
||||||
|
# 前端路由支持(SPA)
|
||||||
|
location / {
|
||||||
|
try_files $uri $uri/ /index.html;
|
||||||
|
}
|
||||||
|
|
||||||
|
# API 代理到后端网关(生产环境)
|
||||||
|
location /api/ {
|
||||||
|
proxy_pass http://gateway:80/;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_connect_timeout 60s;
|
||||||
|
proxy_send_timeout 60s;
|
||||||
|
proxy_read_timeout 60s;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 安全响应头
|
||||||
|
add_header X-Frame-Options "SAMEORIGIN" always;
|
||||||
|
add_header X-Content-Type-Options "nosniff" always;
|
||||||
|
add_header X-XSS-Protection "1; mode=block" always;
|
||||||
|
add_header Referrer-Policy "strict-origin-when-cross-origin" always;
|
||||||
|
}
|
||||||
13
frontend/index.html
Normal file
13
frontend/index.html
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="zh-CN">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<title>Asset Helper 管理后台</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="root"></div>
|
||||||
|
<script type="module" src="/src/main.tsx"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
28
frontend/package.json
Normal file
28
frontend/package.json
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
{
|
||||||
|
"name": "asset-helper-admin",
|
||||||
|
"private": true,
|
||||||
|
"version": "0.1.0",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "tsc -b && vite build",
|
||||||
|
"preview": "vite preview",
|
||||||
|
"lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@ant-design/icons": "^5.6.1",
|
||||||
|
"antd": "^5.24.6",
|
||||||
|
"axios": "^1.8.4",
|
||||||
|
"react": "^18.3.1",
|
||||||
|
"react-dom": "^18.3.1",
|
||||||
|
"react-router-dom": "^6.30.0",
|
||||||
|
"zustand": "^5.0.3"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/react": "^18.3.20",
|
||||||
|
"@types/react-dom": "^18.3.6",
|
||||||
|
"@vitejs/plugin-react": "^4.4.1",
|
||||||
|
"typescript": "~5.7.3",
|
||||||
|
"vite": "^6.3.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
20
frontend/src/App.tsx
Normal file
20
frontend/src/App.tsx
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { RouterProvider } from 'react-router-dom'
|
||||||
|
import { ConfigProvider, theme } from 'antd'
|
||||||
|
import zhCN from 'antd/locale/zh_CN'
|
||||||
|
import { router } from './router'
|
||||||
|
import { useThemeStore } from './stores/theme'
|
||||||
|
|
||||||
|
export default function App() {
|
||||||
|
const isDark = useThemeStore((s) => s.isDark)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ConfigProvider
|
||||||
|
locale={zhCN}
|
||||||
|
theme={{
|
||||||
|
algorithm: isDark ? theme.darkAlgorithm : theme.defaultAlgorithm,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<RouterProvider router={router} />
|
||||||
|
</ConfigProvider>
|
||||||
|
)
|
||||||
|
}
|
||||||
23
frontend/src/api/auth.ts
Normal file
23
frontend/src/api/auth.ts
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { rawPost } from './client'
|
||||||
|
import type { LoginResponse } from '@/types/api'
|
||||||
|
import type { LoginAccountPayload, LoginEmailPayload, RegisterAccountPayload, RegisterEmailPayload } from '@/types/auth'
|
||||||
|
|
||||||
|
/** 账号密码登录 */
|
||||||
|
export function loginAccount(payload: LoginAccountPayload) {
|
||||||
|
return rawPost<LoginAccountPayload, LoginResponse>('/api/v1/auth/login/account', payload)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 邮箱密码登录 */
|
||||||
|
export function loginEmail(payload: LoginEmailPayload) {
|
||||||
|
return rawPost<LoginEmailPayload, LoginResponse>('/api/v1/auth/login/email', payload)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 账号注册 */
|
||||||
|
export function registerAccount(payload: RegisterAccountPayload) {
|
||||||
|
return rawPost<RegisterAccountPayload, LoginResponse>('/api/v1/users/register/account', payload)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 邮箱注册 */
|
||||||
|
export function registerEmail(payload: RegisterEmailPayload) {
|
||||||
|
return rawPost<RegisterEmailPayload, LoginResponse>('/api/v1/users/register/email', payload)
|
||||||
|
}
|
||||||
95
frontend/src/api/client.ts
Normal file
95
frontend/src/api/client.ts
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
import axios, { AxiosError, type AxiosInstance, type AxiosRequestConfig } from 'axios'
|
||||||
|
import { Device, Language, type ApiRequest, type ApiResponse, type ErrorResponse } from '@/types/api'
|
||||||
|
import { useAuthStore } from '@/stores/auth'
|
||||||
|
|
||||||
|
// 根据环境选择基础地址
|
||||||
|
// Docker 开发环境:Vite proxy 会将 /api 转发到后端
|
||||||
|
// 生产环境:Nginx 代理 /api 到后端网关
|
||||||
|
const baseURL = import.meta.env.DEV ? '/' : '/api'
|
||||||
|
|
||||||
|
const client: AxiosInstance = axios.create({
|
||||||
|
baseURL,
|
||||||
|
timeout: 30000,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// 请求拦截器:注入 JWT
|
||||||
|
client.interceptors.request.use((config) => {
|
||||||
|
const token = useAuthStore.getState().token
|
||||||
|
if (token && config.headers) {
|
||||||
|
config.headers.Authorization = `Bearer ${token}`
|
||||||
|
}
|
||||||
|
return config
|
||||||
|
})
|
||||||
|
|
||||||
|
// 响应拦截器:统一错误处理
|
||||||
|
client.interceptors.response.use(
|
||||||
|
(response) => response,
|
||||||
|
(error: AxiosError<ErrorResponse>) => {
|
||||||
|
const status = error.response?.status
|
||||||
|
const message = error.response?.data?.message || error.message || '请求失败'
|
||||||
|
|
||||||
|
if (status === 401) {
|
||||||
|
useAuthStore.getState().logout()
|
||||||
|
window.location.href = '/login'
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.reject(new Error(message))
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
/** 包装业务请求(注册/业务类接口) */
|
||||||
|
export async function apiPost<TReq, TRes>(
|
||||||
|
url: string,
|
||||||
|
data: TReq,
|
||||||
|
config?: AxiosRequestConfig
|
||||||
|
): Promise<TRes> {
|
||||||
|
const body: ApiRequest<TReq> = {
|
||||||
|
device: Device.Web,
|
||||||
|
language: Language.SimplifiedChinese,
|
||||||
|
data,
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await client.post<ApiResponse<TRes>>(url, body, config)
|
||||||
|
const result = response.data
|
||||||
|
|
||||||
|
if (!result.success) {
|
||||||
|
throw new Error(result.message)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.data === null) {
|
||||||
|
throw new Error('接口返回数据为空')
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.data
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 原始 POST(登录/认证类接口,不包装) */
|
||||||
|
export async function rawPost<TReq, TRes>(
|
||||||
|
url: string,
|
||||||
|
data: TReq,
|
||||||
|
config?: AxiosRequestConfig
|
||||||
|
): Promise<TRes> {
|
||||||
|
const response = await client.post<TRes>(url, data, config)
|
||||||
|
return response.data
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 通用 GET */
|
||||||
|
export async function apiGet<TRes>(url: string, config?: AxiosRequestConfig): Promise<TRes> {
|
||||||
|
const response = await client.get<ApiResponse<TRes>>(url, config)
|
||||||
|
const result = response.data
|
||||||
|
|
||||||
|
if (!result.success) {
|
||||||
|
throw new Error(result.message)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.data === null) {
|
||||||
|
throw new Error('接口返回数据为空')
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.data
|
||||||
|
}
|
||||||
|
|
||||||
|
export default client
|
||||||
104
frontend/src/layouts/MainLayout.tsx
Normal file
104
frontend/src/layouts/MainLayout.tsx
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { Outlet, useNavigate } from 'react-router-dom'
|
||||||
|
import { Layout, Menu, Button, theme, Dropdown, Avatar, Space } from 'antd'
|
||||||
|
import {
|
||||||
|
DashboardOutlined,
|
||||||
|
LogoutOutlined,
|
||||||
|
UserOutlined,
|
||||||
|
MenuFoldOutlined,
|
||||||
|
MenuUnfoldOutlined,
|
||||||
|
} from '@ant-design/icons'
|
||||||
|
import { useAuthStore } from '@/stores/auth'
|
||||||
|
|
||||||
|
const { Header, Sider, Content } = Layout
|
||||||
|
|
||||||
|
export default function MainLayout() {
|
||||||
|
const [collapsed, setCollapsed] = useState(false)
|
||||||
|
const logout = useAuthStore((s) => s.logout)
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
const {
|
||||||
|
token: { colorBgContainer, borderRadiusLG },
|
||||||
|
} = theme.useToken()
|
||||||
|
|
||||||
|
const menuItems = [
|
||||||
|
{
|
||||||
|
key: '/',
|
||||||
|
icon: <DashboardOutlined />,
|
||||||
|
label: '仪表盘',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
const userMenuItems = [
|
||||||
|
{
|
||||||
|
key: 'logout',
|
||||||
|
icon: <LogoutOutlined />,
|
||||||
|
label: '退出登录',
|
||||||
|
danger: true,
|
||||||
|
onClick: () => {
|
||||||
|
logout()
|
||||||
|
navigate('/login')
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Layout style={{ minHeight: '100vh' }}>
|
||||||
|
<Sider trigger={null} collapsible collapsed={collapsed} theme="light">
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
height: 64,
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
fontWeight: 'bold',
|
||||||
|
fontSize: collapsed ? 14 : 18,
|
||||||
|
borderBottom: '1px solid #f0f0f0',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{collapsed ? 'AH' : 'Asset Helper'}
|
||||||
|
</div>
|
||||||
|
<Menu
|
||||||
|
mode="inline"
|
||||||
|
defaultSelectedKeys={['/']}
|
||||||
|
items={menuItems}
|
||||||
|
onClick={({ key }) => navigate(key)}
|
||||||
|
/>
|
||||||
|
</Sider>
|
||||||
|
<Layout>
|
||||||
|
<Header
|
||||||
|
style={{
|
||||||
|
padding: '0 24px',
|
||||||
|
background: colorBgContainer,
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'space-between',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
type="text"
|
||||||
|
icon={collapsed ? <MenuUnfoldOutlined /> : <MenuFoldOutlined />}
|
||||||
|
onClick={() => setCollapsed(!collapsed)}
|
||||||
|
/>
|
||||||
|
<Dropdown menu={{ items: userMenuItems }} placement="bottomRight">
|
||||||
|
<Space style={{ cursor: 'pointer' }}>
|
||||||
|
<Avatar icon={<UserOutlined />} />
|
||||||
|
<span>管理员</span>
|
||||||
|
</Space>
|
||||||
|
</Dropdown>
|
||||||
|
</Header>
|
||||||
|
<Content
|
||||||
|
style={{
|
||||||
|
margin: 24,
|
||||||
|
padding: 24,
|
||||||
|
background: colorBgContainer,
|
||||||
|
borderRadius: borderRadiusLG,
|
||||||
|
overflow: 'auto',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Outlet />
|
||||||
|
</Content>
|
||||||
|
</Layout>
|
||||||
|
</Layout>
|
||||||
|
)
|
||||||
|
}
|
||||||
10
frontend/src/main.tsx
Normal file
10
frontend/src/main.tsx
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { StrictMode } from 'react'
|
||||||
|
import { createRoot } from 'react-dom/client'
|
||||||
|
import './styles/global.css'
|
||||||
|
import App from './App'
|
||||||
|
|
||||||
|
createRoot(document.getElementById('root')!).render(
|
||||||
|
<StrictMode>
|
||||||
|
<App />
|
||||||
|
</StrictMode>
|
||||||
|
)
|
||||||
35
frontend/src/pages/DashboardPage.tsx
Normal file
35
frontend/src/pages/DashboardPage.tsx
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import { Card, Statistic, Row, Col } from 'antd'
|
||||||
|
import { UserOutlined, AppstoreOutlined } from '@ant-design/icons'
|
||||||
|
|
||||||
|
export default function DashboardPage() {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<h2 style={{ marginBottom: 24 }}>仪表盘</h2>
|
||||||
|
<Row gutter={16}>
|
||||||
|
<Col span={8}>
|
||||||
|
<Card>
|
||||||
|
<Statistic
|
||||||
|
title="用户总数"
|
||||||
|
value={0}
|
||||||
|
prefix={<UserOutlined />}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
</Col>
|
||||||
|
<Col span={8}>
|
||||||
|
<Card>
|
||||||
|
<Statistic
|
||||||
|
title="资产总数"
|
||||||
|
value={0}
|
||||||
|
prefix={<AppstoreOutlined />}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
</Col>
|
||||||
|
<Col span={8}>
|
||||||
|
<Card>
|
||||||
|
<Statistic title="今日新增" value={0} />
|
||||||
|
</Card>
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
125
frontend/src/pages/LoginPage.tsx
Normal file
125
frontend/src/pages/LoginPage.tsx
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
import { useState, useEffect } from 'react'
|
||||||
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
import { Card, Form, Input, Button, Tabs, message } from 'antd'
|
||||||
|
import { UserOutlined, LockOutlined, MailOutlined, MoonOutlined, SunOutlined } from '@ant-design/icons'
|
||||||
|
import { loginAccount, loginEmail } from '@/api/auth'
|
||||||
|
import { useAuthStore } from '@/stores/auth'
|
||||||
|
import { useThemeStore } from '@/stores/theme'
|
||||||
|
import type { LoginAccountPayload, LoginEmailPayload } from '@/types/auth'
|
||||||
|
|
||||||
|
type LoginType = 'account' | 'email'
|
||||||
|
|
||||||
|
export default function LoginPage() {
|
||||||
|
const [loginType, setLoginType] = useState<LoginType>('account')
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const login = useAuthStore((s) => s.login)
|
||||||
|
const isDark = useThemeStore((s) => s.isDark)
|
||||||
|
const toggleTheme = useThemeStore((s) => s.toggle)
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const htmlOverflow = document.documentElement.style.overflow
|
||||||
|
const bodyOverflow = document.body.style.overflow
|
||||||
|
const bodyTouchAction = document.body.style.touchAction
|
||||||
|
|
||||||
|
document.documentElement.style.overflow = 'hidden'
|
||||||
|
document.body.style.overflow = 'hidden'
|
||||||
|
document.body.style.touchAction = 'none'
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
document.documentElement.style.overflow = htmlOverflow
|
||||||
|
document.body.style.overflow = bodyOverflow
|
||||||
|
document.body.style.touchAction = bodyTouchAction
|
||||||
|
}
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
async function handleSubmit(values: LoginAccountPayload | LoginEmailPayload) {
|
||||||
|
setLoading(true)
|
||||||
|
try {
|
||||||
|
const response =
|
||||||
|
loginType === 'account'
|
||||||
|
? await loginAccount(values as LoginAccountPayload)
|
||||||
|
: await loginEmail(values as LoginEmailPayload)
|
||||||
|
|
||||||
|
if (response.success && response.token) {
|
||||||
|
login(response.token)
|
||||||
|
message.success('登录成功')
|
||||||
|
navigate('/')
|
||||||
|
} else {
|
||||||
|
message.error(response.message || '登录失败')
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
message.error(err instanceof Error ? err.message : '登录失败')
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
height: '100vh',
|
||||||
|
overflow: 'hidden',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
background: isDark ? '#141414' : '#f0f2f5',
|
||||||
|
transition: 'background 0.3s ease',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Card
|
||||||
|
title="Asset Helper 管理后台"
|
||||||
|
extra={
|
||||||
|
<Button
|
||||||
|
type="text"
|
||||||
|
icon={isDark ? <SunOutlined /> : <MoonOutlined />}
|
||||||
|
onClick={toggleTheme}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
style={{ width: 400 }}
|
||||||
|
styles={{ header: { textAlign: 'center', fontSize: 18 } }}
|
||||||
|
>
|
||||||
|
<Tabs
|
||||||
|
centered
|
||||||
|
activeKey={loginType}
|
||||||
|
onChange={(key) => setLoginType(key as LoginType)}
|
||||||
|
items={[
|
||||||
|
{ key: 'account', label: '账号登录' },
|
||||||
|
{ key: 'email', label: '邮箱登录' },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<Form onFinish={handleSubmit} autoComplete="off">
|
||||||
|
{loginType === 'account' ? (
|
||||||
|
<Form.Item
|
||||||
|
name="account"
|
||||||
|
rules={[{ required: true, message: '请输入账号' }]}
|
||||||
|
>
|
||||||
|
<Input prefix={<UserOutlined />} placeholder="账号" />
|
||||||
|
</Form.Item>
|
||||||
|
) : (
|
||||||
|
<Form.Item
|
||||||
|
name="email"
|
||||||
|
rules={[
|
||||||
|
{ required: true, message: '请输入邮箱' },
|
||||||
|
{ type: 'email', message: '邮箱格式不正确' },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input prefix={<MailOutlined />} placeholder="邮箱" />
|
||||||
|
</Form.Item>
|
||||||
|
)}
|
||||||
|
<Form.Item
|
||||||
|
name="password"
|
||||||
|
rules={[{ required: true, message: '请输入密码' }]}
|
||||||
|
>
|
||||||
|
<Input.Password prefix={<LockOutlined />} placeholder="密码" />
|
||||||
|
</Form.Item>
|
||||||
|
<Form.Item>
|
||||||
|
<Button type="primary" htmlType="submit" block loading={loading}>
|
||||||
|
登录
|
||||||
|
</Button>
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
19
frontend/src/pages/NotFoundPage.tsx
Normal file
19
frontend/src/pages/NotFoundPage.tsx
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import { Button, Result } from 'antd'
|
||||||
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
|
||||||
|
export default function NotFoundPage() {
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Result
|
||||||
|
status="404"
|
||||||
|
title="404"
|
||||||
|
subTitle="抱歉,您访问的页面不存在"
|
||||||
|
extra={
|
||||||
|
<Button type="primary" onClick={() => navigate('/')}>
|
||||||
|
返回首页
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
38
frontend/src/router/index.tsx
Normal file
38
frontend/src/router/index.tsx
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
import { createBrowserRouter, Navigate, Outlet } from 'react-router-dom'
|
||||||
|
import { useAuthStore } from '@/stores/auth'
|
||||||
|
import LoginPage from '@/pages/LoginPage'
|
||||||
|
import DashboardPage from '@/pages/DashboardPage'
|
||||||
|
import NotFoundPage from '@/pages/NotFoundPage'
|
||||||
|
import MainLayout from '@/layouts/MainLayout'
|
||||||
|
|
||||||
|
/** 路由守卫:已登录则放行,未登录跳转登录页 */
|
||||||
|
function RequireAuth() {
|
||||||
|
const isAuthenticated = useAuthStore((s) => s.isAuthenticated)
|
||||||
|
return isAuthenticated ? <Outlet /> : <Navigate to="/login" replace />
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 登录页守卫:已登录则跳转首页 */
|
||||||
|
function LoginGuard() {
|
||||||
|
const isAuthenticated = useAuthStore((s) => s.isAuthenticated)
|
||||||
|
return isAuthenticated ? <Navigate to="/" replace /> : <LoginPage />
|
||||||
|
}
|
||||||
|
|
||||||
|
export const router = createBrowserRouter([
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
element: <LoginGuard />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
element: <RequireAuth />,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: '/',
|
||||||
|
element: <MainLayout />,
|
||||||
|
children: [
|
||||||
|
{ index: true, element: <DashboardPage /> },
|
||||||
|
{ path: '*', element: <NotFoundPage /> },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
])
|
||||||
25
frontend/src/stores/auth.ts
Normal file
25
frontend/src/stores/auth.ts
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import { create } from 'zustand'
|
||||||
|
import { devtools } from 'zustand/middleware'
|
||||||
|
|
||||||
|
interface AuthState {
|
||||||
|
token: string | null
|
||||||
|
isAuthenticated: boolean
|
||||||
|
login: (token: string) => void
|
||||||
|
logout: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useAuthStore = create<AuthState>()(
|
||||||
|
devtools(
|
||||||
|
(set) => ({
|
||||||
|
token: null,
|
||||||
|
isAuthenticated: false,
|
||||||
|
|
||||||
|
login: (token: string) =>
|
||||||
|
set({ token, isAuthenticated: true }, false, 'login'),
|
||||||
|
|
||||||
|
logout: () =>
|
||||||
|
set({ token: null, isAuthenticated: false }, false, 'logout'),
|
||||||
|
}),
|
||||||
|
{ name: 'AuthStore' }
|
||||||
|
)
|
||||||
|
)
|
||||||
20
frontend/src/stores/theme.ts
Normal file
20
frontend/src/stores/theme.ts
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { create } from 'zustand'
|
||||||
|
import { devtools, persist } from 'zustand/middleware'
|
||||||
|
|
||||||
|
interface ThemeState {
|
||||||
|
isDark: boolean
|
||||||
|
toggle: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useThemeStore = create<ThemeState>()(
|
||||||
|
devtools(
|
||||||
|
persist(
|
||||||
|
(set) => ({
|
||||||
|
isDark: false,
|
||||||
|
toggle: () => set((state) => ({ isDark: !state.isDark }), false, 'toggleTheme'),
|
||||||
|
}),
|
||||||
|
{ name: 'theme-storage' }
|
||||||
|
),
|
||||||
|
{ name: 'ThemeStore' }
|
||||||
|
)
|
||||||
|
)
|
||||||
5
frontend/src/styles/global.css
Normal file
5
frontend/src/styles/global.css
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
transition: background-color 0.3s ease;
|
||||||
|
}
|
||||||
47
frontend/src/types/api.ts
Normal file
47
frontend/src/types/api.ts
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
// 与后端对齐的通用 API 类型
|
||||||
|
|
||||||
|
/** device 编码 */
|
||||||
|
export enum Device {
|
||||||
|
IOS = 1,
|
||||||
|
Android = 2,
|
||||||
|
Web = 3,
|
||||||
|
IPad = 4,
|
||||||
|
MacOS = 5,
|
||||||
|
Windows = 6,
|
||||||
|
Linux = 7,
|
||||||
|
}
|
||||||
|
|
||||||
|
/** language 编码 */
|
||||||
|
export enum Language {
|
||||||
|
SimplifiedChinese = 1,
|
||||||
|
TraditionalChinese = 2,
|
||||||
|
English = 3,
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 注册/业务类接口请求包装 */
|
||||||
|
export interface ApiRequest<T> {
|
||||||
|
device: number
|
||||||
|
language: number
|
||||||
|
data: T
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 注册/业务类接口响应包装 */
|
||||||
|
export interface ApiResponse<T> {
|
||||||
|
success: boolean
|
||||||
|
message: string
|
||||||
|
data: T | null
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 登录/认证类接口响应 */
|
||||||
|
export interface LoginResponse {
|
||||||
|
success: boolean
|
||||||
|
token: string | null
|
||||||
|
message: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 网关错误响应(HTTP 非 200) */
|
||||||
|
export interface ErrorResponse {
|
||||||
|
error: string
|
||||||
|
message: string
|
||||||
|
code: number
|
||||||
|
}
|
||||||
27
frontend/src/types/auth.ts
Normal file
27
frontend/src/types/auth.ts
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
// 认证相关类型
|
||||||
|
|
||||||
|
export interface LoginAccountPayload {
|
||||||
|
account: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LoginEmailPayload {
|
||||||
|
email: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RegisterAccountPayload {
|
||||||
|
account: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RegisterEmailPayload {
|
||||||
|
email: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface UserProfile {
|
||||||
|
id: string
|
||||||
|
account?: string
|
||||||
|
email?: string
|
||||||
|
}
|
||||||
29
frontend/src/utils/storage.ts
Normal file
29
frontend/src/utils/storage.ts
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
// 轻量存储封装(JWT 存内存,其他持久化数据用 localStorage)
|
||||||
|
|
||||||
|
const PREFIX = 'asset_helper_admin:'
|
||||||
|
|
||||||
|
export const storage = {
|
||||||
|
get<T>(key: string): T | null {
|
||||||
|
try {
|
||||||
|
const raw = localStorage.getItem(PREFIX + key)
|
||||||
|
return raw ? (JSON.parse(raw) as T) : null
|
||||||
|
} catch {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
set<T>(key: string, value: T): void {
|
||||||
|
localStorage.setItem(PREFIX + key, JSON.stringify(value))
|
||||||
|
},
|
||||||
|
|
||||||
|
remove(key: string): void {
|
||||||
|
localStorage.removeItem(PREFIX + key)
|
||||||
|
},
|
||||||
|
|
||||||
|
clear(): void {
|
||||||
|
// 仅清除本应用前缀的 key
|
||||||
|
Object.keys(localStorage)
|
||||||
|
.filter((k) => k.startsWith(PREFIX))
|
||||||
|
.forEach((k) => localStorage.removeItem(k))
|
||||||
|
},
|
||||||
|
}
|
||||||
1
frontend/src/vite-env.d.ts
vendored
Normal file
1
frontend/src/vite-env.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
/// <reference types="vite/client" />
|
||||||
25
frontend/tsconfig.app.json
Normal file
25
frontend/tsconfig.app.json
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
||||||
|
"target": "ES2020",
|
||||||
|
"useDefineForClassFields": true,
|
||||||
|
"lib": ["ES2020", "DOM", "DOM.Iterable"],
|
||||||
|
"module": "ESNext",
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"moduleResolution": "bundler",
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
|
"isolatedModules": true,
|
||||||
|
"moduleDetection": "force",
|
||||||
|
"noEmit": true,
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"strict": true,
|
||||||
|
"noUnusedLocals": true,
|
||||||
|
"noUnusedParameters": true,
|
||||||
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"baseUrl": ".",
|
||||||
|
"paths": {
|
||||||
|
"@/*": ["src/*"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"include": ["src"]
|
||||||
|
}
|
||||||
7
frontend/tsconfig.json
Normal file
7
frontend/tsconfig.json
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"files": [],
|
||||||
|
"references": [
|
||||||
|
{ "path": "./tsconfig.app.json" },
|
||||||
|
{ "path": "./tsconfig.node.json" }
|
||||||
|
]
|
||||||
|
}
|
||||||
19
frontend/tsconfig.node.json
Normal file
19
frontend/tsconfig.node.json
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
|
||||||
|
"target": "ES2022",
|
||||||
|
"lib": ["ES2023"],
|
||||||
|
"module": "ESNext",
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"moduleResolution": "bundler",
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
|
"isolatedModules": true,
|
||||||
|
"moduleDetection": "force",
|
||||||
|
"noEmit": true,
|
||||||
|
"strict": true,
|
||||||
|
"noUnusedLocals": true,
|
||||||
|
"noUnusedParameters": true,
|
||||||
|
"noFallthroughCasesInSwitch": true
|
||||||
|
},
|
||||||
|
"include": ["vite.config.ts"]
|
||||||
|
}
|
||||||
32
frontend/vite.config.ts
Normal file
32
frontend/vite.config.ts
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import { defineConfig } from 'vite'
|
||||||
|
import react from '@vitejs/plugin-react'
|
||||||
|
import path from 'path'
|
||||||
|
|
||||||
|
export default defineConfig(({ mode }) => ({
|
||||||
|
plugins: [react()],
|
||||||
|
resolve: {
|
||||||
|
alias: {
|
||||||
|
'@': path.resolve(__dirname, './src'),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
server: {
|
||||||
|
host: '0.0.0.0',
|
||||||
|
port: 5173,
|
||||||
|
strictPort: true,
|
||||||
|
watch: {
|
||||||
|
usePolling: true,
|
||||||
|
},
|
||||||
|
proxy: {
|
||||||
|
'/api': {
|
||||||
|
// 开发环境:通过 Docker 网络直接访问网关容器
|
||||||
|
target: process.env.VITE_API_BASE_URL || 'http://api-gateway',
|
||||||
|
changeOrigin: true,
|
||||||
|
secure: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
build: {
|
||||||
|
outDir: 'dist',
|
||||||
|
sourcemap: mode !== 'production',
|
||||||
|
},
|
||||||
|
}))
|
||||||
Reference in New Issue
Block a user