Update API and WebSocket base URLs to use environment variables for better configuration management

This commit is contained in:
liu
2026-01-30 00:45:04 +08:00
parent 197ce5e7ea
commit 246515b43e
11 changed files with 310 additions and 2 deletions

42
.dockerignore Normal file
View File

@@ -0,0 +1,42 @@
# Git
.git
.gitignore
.gitattributes
# 文档
README.md
LICENSE
开发文档/
# IDE
.vscode/
.idea/
*.iml
*.swp
*.swo
# 构建产物Docker 内会重新构建)
frontend/node_modules/
frontend/dist/
backend/target/
*.jar
*.war
# 日志
*.log
npm-debug.log*
yarn-debug.log*
# 系统文件
.DS_Store
Thumbs.db
Desktop.ini
# Docker 相关(避免递归)
docker/
.dockerignore
# 其他
*.local
.cache/
coverage/

View File

@@ -0,0 +1,34 @@
package com.music.config;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.Resource;
import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import org.springframework.web.servlet.resource.PathResourceResolver;
import java.io.IOException;
/**
* 生产环境:前端 SPA 由 Spring Boot 托管时,未匹配到的路径回退到 index.html。
*/
@Configuration
public class SpaForwardConfig implements WebMvcConfigurer {
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler("/**")
.addResourceLocations("classpath:/static/")
.resourceChain(true)
.addResolver(new PathResourceResolver() {
@Override
protected Resource getResource(String resourcePath, Resource location) throws IOException {
Resource resource = location.createRelative(resourcePath);
if (resource.exists() && resource.isReadable()) {
return resource;
}
// SPA 路由回退到 index.html
return location.createRelative("index.html");
}
});
}
}

44
docker/Dockerfile Normal file
View File

@@ -0,0 +1,44 @@
# MangTool 前后端单容器构建
# 阶段1构建前端
FROM node:20-alpine AS frontend-builder
WORKDIR /app/frontend
COPY frontend/package.json frontend/package-lock.json ./
RUN npm ci --ignore-scripts
COPY frontend/ ./
# 生产环境使用相对路径,与后端同源
ENV VITE_API_BASE_URL=
RUN npm run build
# 阶段2构建后端含前端静态资源
FROM maven:3.8-eclipse-temurin-8-alpine AS backend-builder
WORKDIR /app
# 复制后端源码
COPY backend/pom.xml ./
RUN mvn dependency:go-offline -B
COPY backend/ ./
# 从阶段1 复制前端构建产物到 Spring Boot 静态目录
COPY --from=frontend-builder /app/frontend/dist ./src/main/resources/static
RUN mvn package -DskipTests -B
# 阶段3运行镜像
FROM eclipse-temurin:8-jre-alpine
WORKDIR /app
RUN apk add --no-cache tzdata wget \
&& cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
&& echo "Asia/Shanghai" > /etc/timezone
COPY --from=backend-builder /app/target/*.jar app.jar
EXPOSE 8080
# 健康检查
HEALTHCHECK --interval=30s --timeout=3s --start-period=40s --retries=3 \
CMD wget --quiet --tries=1 --spider http://localhost:8080/api/health || exit 1
ENTRYPOINT ["java", "-jar", "app.jar"]

75
docker/README.md Normal file
View File

@@ -0,0 +1,75 @@
# MangTool Docker 部署
前后端单容器部署:拉代码后进入 `docker` 目录,执行启动脚本即可。
## 环境要求
- Docker
- Docker Composev2 推荐:`docker compose`
## 部署步骤
### 1. 拉取代码
```bash
git clone <仓库地址>
cd MyTool
```
### 2. 启动服务
**Linux / macOS**
```bash
cd docker
chmod +x start.sh
./start.sh
```
**Windows**
在资源管理器中进入 `docker` 目录,双击运行 `start.bat`;或在终端执行:
```cmd
cd docker
start.bat
```
或直接使用 docker compose
```bash
cd docker
docker compose up -d --build
```
### 3. 访问应用
浏览器打开:**http://localhost:8080**
前端与后端由同一服务提供,无需单独配置 API 地址。
## 常用命令
| 操作 | 命令 |
|------------|------------------------------|
| 后台启动 | `docker compose up -d --build` |
| 查看日志 | `docker compose logs -f` |
| 停止并删除 | `docker compose down` |
| 仅重新构建 | `docker compose build --no-cache` |
| 查看状态 | `docker compose ps` |
| 健康检查 | `docker compose exec mangtool wget -q -O- http://localhost:8080/api/health` |
## 端口与数据
- **端口**:宿主机 `8080` 映射容器 `8080`,可在 `docker-compose.yml` 中修改左侧端口,例如 `"8888:8080"`
- **数据**:工具读写路径在容器内;若需挂载宿主机目录(如音乐库、输入输出目录),在 `docker-compose.yml` 中取消 `volumes` 注释并改为实际路径。
## 构建说明
- **Dockerfile**:多阶段构建
1. 使用 Node 20 构建前端Vite产出到 `dist`
2. 使用 Maven + JDK 8 构建后端,并将前端 `dist` 拷贝到 `src/main/resources/static`
3. 运行阶段使用 `eclipse-temurin:8-jre-alpine`,仅运行打包好的 Spring Boot jar
4. 内置健康检查(每 30 秒检查 `/api/health` 端点)
- 生产环境前端 API/WebSocket 使用相对路径,与后端同源,无需再配 CORS。
- 已配置 `.dockerignore` 优化构建上下文,加快构建速度。

16
docker/docker-compose.yml Normal file
View File

@@ -0,0 +1,16 @@
version: "3.8"
services:
mangtool:
build:
context: ..
dockerfile: docker/Dockerfile
image: mangtool:latest
container_name: mangtool
ports:
- "8080:8080"
restart: unless-stopped
# 如需挂载本地目录给工具读写,可取消注释并修改路径
# volumes:
# - /path/on/host/Input:/app/data/Input
# - /path/on/host/Library_Final:/app/data/Library_Final

35
docker/start.bat Normal file
View File

@@ -0,0 +1,35 @@
@echo off
chcp 65001 >nul
cd /d "%~dp0"
echo 检查 Docker 状态...
docker info >nul 2>&1
if errorlevel 1 (
echo ❌ 错误: Docker 未运行,请先启动 Docker Desktop
pause
exit /b 1
)
echo.
echo ^>^>^> MangTool Docker 启动中...
docker compose up -d --build
if errorlevel 1 (
echo.
echo ❌ 启动失败,请检查日志: docker compose logs
pause
exit /b 1
)
echo.
echo ✅ 启动完成!
echo.
echo 📌 访问地址: http://localhost:8080
echo 📌 查看日志: docker compose logs -f
echo 📌 停止服务: docker compose down
echo.
echo 等待服务就绪中...
timeout /t 3 /nobreak >nul
docker compose ps
echo.
pause

29
docker/start.sh Normal file
View File

@@ -0,0 +1,29 @@
#!/usr/bin/env bash
# MangTool 一键启动脚本(拉代码后在此目录执行)
set -e
cd "$(dirname "$0")"
# 检查 Docker 是否运行
if ! docker info > /dev/null 2>&1; then
echo "❌ 错误: Docker 未运行,请先启动 Docker"
exit 1
fi
echo ">>> MangTool Docker 启动中..."
docker compose up -d --build
if [ $? -eq 0 ]; then
echo ""
echo "✅ 启动完成!"
echo ""
echo "📌 访问地址: http://localhost:8080"
echo "📌 查看日志: docker compose logs -f"
echo "📌 停止服务: docker compose down"
echo ""
echo "等待服务就绪中..."
sleep 3
docker compose ps
else
echo "❌ 启动失败,请检查日志: docker compose logs"
exit 1
fi

15
docker/stop.bat Normal file
View File

@@ -0,0 +1,15 @@
@echo off
chcp 65001 >nul
cd /d "%~dp0"
echo 正在停止 MangTool 容器...
docker compose down
if errorlevel 1 (
echo ❌ 停止失败
pause
exit /b 1
)
echo ✅ 已停止并删除容器
pause

14
docker/stop.sh Normal file
View File

@@ -0,0 +1,14 @@
#!/usr/bin/env bash
# MangTool Docker 停止脚本
set -e
cd "$(dirname "$0")"
echo ">>> 正在停止 MangTool 容器..."
docker compose down
if [ $? -eq 0 ]; then
echo "✅ 已停止并删除容器"
else
echo "❌ 停止失败"
exit 1
fi

View File

@@ -2,7 +2,7 @@ import axios from 'axios';
import { ElMessage } from 'element-plus'; import { ElMessage } from 'element-plus';
const request = axios.create({ const request = axios.create({
baseURL: 'http://localhost:8080', baseURL: import.meta.env.VITE_API_BASE_URL ?? 'http://localhost:8080',
timeout: 30000, timeout: 30000,
headers: { headers: {
'Content-Type': 'application/json' 'Content-Type': 'application/json'

View File

@@ -30,7 +30,11 @@ export function useWebSocket(taskId: string | null, onMessage: (msg: ProgressMes
} }
try { try {
const socket = new SockJS('http://localhost:8080/ws'); const wsBase =
import.meta.env.VITE_WS_BASE_URL !== undefined && import.meta.env.VITE_WS_BASE_URL !== ''
? import.meta.env.VITE_WS_BASE_URL
: (typeof window !== 'undefined' ? window.location.origin : 'http://localhost:8080');
const socket = new SockJS(`${wsBase}/ws`);
stompClient = Stomp.over(socket); stompClient = Stomp.over(socket);
// 禁用调试日志 // 禁用调试日志