938 lines
23 KiB
Markdown
938 lines
23 KiB
Markdown
# 域名部署实施计划
|
|
|
|
> **For agentic workers:** REQUIRED: Use superpowers:subagent-driven-development (if subagents available) or superpowers:executing-plans to implement this plan. Steps use checkbox (`- [ ]`) syntax for tracking.
|
|
|
|
**Goal:** 将情绪博物馆所有服务迁移到域名 lifescript.happylifeos.com,配置 HTTPS SSL 证书,并实现自动续期。
|
|
|
|
**Architecture:**
|
|
- 前端应用通过 Nginx 托管,使用相对路径 API 调用
|
|
- 后端 API 通过 Nginx 反向代理暴露
|
|
- WebSocket 通过 Nginx 升级协议代理
|
|
- SSL 证书使用 Let's Encrypt 通过 certbot 申请,配置 systemd timer 自动续期
|
|
|
|
**Tech Stack:** Spring Boot, Vue 3, React, Nginx, Let's Encrypt certbot, Python 3
|
|
|
|
---
|
|
|
|
## Chunk 1: SSL 证书申请脚本
|
|
|
|
### Task 1: 创建 SSL 证书申请 Python 脚本
|
|
|
|
**Files:**
|
|
- Create: `tools/deploy-ssl-cert.py`
|
|
|
|
- [ ] **Step 1: 创建 Python 脚本文件**
|
|
|
|
```python
|
|
#!/usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
"""
|
|
SSL 证书申请与自动续期脚本
|
|
使用 certbot 工具申请 Let's Encrypt 证书并配置 nginx
|
|
"""
|
|
|
|
import argparse
|
|
import subprocess
|
|
import sys
|
|
from pathlib import Path
|
|
|
|
DOMAIN = "lifescript.happylifeos.com"
|
|
CERT_PATH = f"/etc/letsencrypt/live/{DOMAIN}/fullchain.pem"
|
|
KEY_PATH = f"/etc/letsencrypt/live/{DOMAIN}/privkey.pem"
|
|
|
|
|
|
def log_info(msg):
|
|
print(f"\033[32m[INFO]\033[0m {msg}")
|
|
|
|
|
|
def log_warn(msg):
|
|
print(f"\033[33m[WARN]\033[0m {msg}")
|
|
|
|
|
|
def log_error(msg):
|
|
print(f"\033[31m[ERROR]\033[0m {msg}")
|
|
|
|
|
|
def run_command(cmd, check=True):
|
|
"""执行 shell 命令"""
|
|
try:
|
|
result = subprocess.run(
|
|
cmd, shell=True, check=check, capture_output=False, text=True
|
|
)
|
|
return result.returncode == 0
|
|
except subprocess.CalledProcessError as e:
|
|
log_error(f"命令执行失败:{e}")
|
|
return False
|
|
|
|
|
|
def check_certbot():
|
|
"""检查 certbot 是否安装"""
|
|
log_info("检查 certbot 是否安装...")
|
|
if not run_command("which certbot", check=False):
|
|
log_error("certbot 未安装,请先安装:apt install certbot python3-certbot-nginx")
|
|
return False
|
|
log_info("certbot 已安装")
|
|
return True
|
|
|
|
|
|
def check_nginx():
|
|
"""检查 nginx 配置是否正确"""
|
|
log_info("检查 nginx 配置...")
|
|
if not run_command("nginx -t", check=False):
|
|
log_error("nginx 配置有误,请先修复")
|
|
return False
|
|
log_info("nginx 配置正确")
|
|
return True
|
|
|
|
|
|
def apply_certificate(dry_run=False):
|
|
"""申请 SSL 证书"""
|
|
log_info(f"开始为 {DOMAIN} 申请 SSL 证书...")
|
|
|
|
cmd = f"certbot --nginx -d {DOMAIN} --non-interactive --agree-tos --email admin@happylifeos.com"
|
|
if dry_run:
|
|
cmd += " --dry-run"
|
|
log_info(f"[DRY RUN] 执行:{cmd}")
|
|
|
|
if run_command(cmd):
|
|
log_info("证书申请成功")
|
|
return True
|
|
else:
|
|
log_error("证书申请失败")
|
|
return False
|
|
|
|
|
|
def setup_auto_renewal():
|
|
"""配置自动续期"""
|
|
log_info("配置证书自动续期...")
|
|
|
|
# 检查是否已存在 systemd timer
|
|
if run_command("systemctl list-timers | grep certbot", check=False):
|
|
log_info("certbot 自动续期 timer 已存在")
|
|
return True
|
|
|
|
# 启用 systemd timer
|
|
if run_command("systemctl enable certbot.timer && systemctl start certbot.timer"):
|
|
log_info("已启用 certbot 自动续期 timer")
|
|
return True
|
|
else:
|
|
log_warn("systemd timer 配置失败,尝试配置 cron 任务")
|
|
return setup_cron_renewal()
|
|
|
|
|
|
def setup_cron_renewal():
|
|
"""配置 cron 自动续期"""
|
|
cron_job = "0 3 1 * * certbot renew --quiet --deploy-hook 'systemctl reload nginx'"
|
|
log_info(f"添加 cron 任务:{cron_job}")
|
|
|
|
# 添加到 crontab
|
|
result = subprocess.run(
|
|
f"(crontab -l 2>/dev/null | grep -v '{DOMAIN}'; echo '{cron_job}') | crontab -",
|
|
shell=True, capture_output=True, text=True
|
|
)
|
|
|
|
if result.returncode == 0:
|
|
log_info("cron 任务添加成功")
|
|
return True
|
|
else:
|
|
log_error("cron 任务添加失败")
|
|
return False
|
|
|
|
|
|
def verify_certificate():
|
|
"""验证证书是否有效"""
|
|
log_info("验证 SSL 证书...")
|
|
cmd = f'echo | openssl s_client -connect {DOMAIN}:443 -servername {DOMAIN} 2>/dev/null | openssl x509 -noout -dates'
|
|
|
|
result = subprocess.run(cmd, shell=True, capture_output=True, text=True)
|
|
if result.returncode == 0:
|
|
log_info(f"证书信息:\n{result.stdout}")
|
|
return True
|
|
else:
|
|
log_warn("无法验证证书(可能域名 DNS 尚未生效)")
|
|
return False
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(description="SSL 证书申请与自动续期脚本")
|
|
parser.add_argument("--dry-run", action="store_true", help="模拟运行,不实际申请证书")
|
|
parser.add_argument("--renew", action="store_true", help="手动续期证书")
|
|
parser.add_argument("--verify", action="store_true", help="验证证书状态")
|
|
parser.add_argument("--setup-renewal", action="store_true", help="仅配置自动续期")
|
|
args = parser.parse_args()
|
|
|
|
# 验证模式
|
|
if args.verify:
|
|
verify_certificate()
|
|
return 0
|
|
|
|
# 仅配置自动续期
|
|
if args.setup_renewal:
|
|
setup_auto_renewal()
|
|
return 0
|
|
|
|
# 手动续期
|
|
if args.renew:
|
|
run_command("certbot renew --force-renewal")
|
|
run_command("systemctl reload nginx")
|
|
return 0
|
|
|
|
# 完整流程
|
|
log_info("=== SSL 证书申请流程 ===")
|
|
|
|
if not check_certbot():
|
|
return 1
|
|
|
|
if not check_nginx():
|
|
return 1
|
|
|
|
if not apply_certificate(dry_run=args.dry_run):
|
|
return 1
|
|
|
|
if not args.dry_run:
|
|
setup_auto_renewal()
|
|
verify_certificate()
|
|
|
|
log_info("=== SSL 证书申请完成 ===")
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
sys.exit(main())
|
|
```
|
|
|
|
- [ ] **Step 2: 添加文件执行权限**
|
|
|
|
```bash
|
|
chmod +x tools/deploy-ssl-cert.py
|
|
```
|
|
|
|
- [ ] **Step 3: 添加 Python 脚本元信息**
|
|
|
|
在文件顶部注释中添加:
|
|
```python
|
|
"""
|
|
Author: [创建人姓名]
|
|
Created: 2026-03-17
|
|
Purpose: SSL 证书申请与自动续期 - 使用 certbot 配置 Let's Encrypt 证书
|
|
"""
|
|
```
|
|
|
|
- [ ] **Step 4: 提交**
|
|
|
|
```bash
|
|
git add tools/deploy-ssl-cert.py
|
|
git commit -m "feat: 添加 SSL 证书申请与自动续期脚本"
|
|
```
|
|
|
|
---
|
|
|
|
## Chunk 2: 前端环境配置文件修改
|
|
|
|
### Task 2: 修改用户前端 (web) 生产环境配置
|
|
|
|
**Files:**
|
|
- Modify: `web/.env.production`
|
|
|
|
- [ ] **Step 1: 读取当前文件内容**
|
|
|
|
- [ ] **Step 2: 修改为域名配置**
|
|
|
|
```env
|
|
# 生产环境配置
|
|
VITE_APP_ENV=prod
|
|
VITE_APP_TITLE=情绪博物馆
|
|
VITE_APP_VERSION=1.0.0
|
|
|
|
# API 配置 - 使用相对路径,通过 nginx 代理
|
|
VITE_API_BASE_URL=/api
|
|
VITE_WS_BASE_URL=/ws
|
|
VITE_UPLOAD_URL=/api/upload
|
|
|
|
# 调试配置
|
|
VITE_DEBUG=false
|
|
VITE_MOCK=false
|
|
|
|
# 其他配置
|
|
VITE_APP_DESCRIPTION=情绪博物馆 Web 系统
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add web/.env.production
|
|
git commit -m "config: 更新 web 前端生产环境配置为域名访问"
|
|
```
|
|
|
|
### Task 3: 修改管理后台 (web-admin) 生产环境配置
|
|
|
|
**Files:**
|
|
- Modify: `web-admin/.env.production`
|
|
|
|
- [ ] **Step 1: 读取当前文件内容**
|
|
|
|
- [ ] **Step 2: 修改为域名配置**
|
|
|
|
```env
|
|
# 生产环境配置
|
|
VITE_APP_TITLE=情绪博物馆管理后台
|
|
# 生产环境使用相对路径,通过 nginx 代理到后端服务
|
|
VITE_APP_BASE_API=/api
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add web-admin/.env.production
|
|
git commit -m "config: 更新管理后台生产环境配置为域名访问"
|
|
```
|
|
|
|
### Task 4: 修改 Life-Script 生产环境配置
|
|
|
|
**Files:**
|
|
- Modify: `life-script/.env.production`
|
|
|
|
- [ ] **Step 1: 读取当前文件内容**
|
|
|
|
- [ ] **Step 2: 修改为域名配置**
|
|
|
|
```env
|
|
# 生产环境配置
|
|
VITE_API_BASE_URL=/api
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add life-script/.env.production
|
|
git commit -m "config: 更新 Life-Script 生产环境配置为域名访问"
|
|
```
|
|
|
|
---
|
|
|
|
## Chunk 3: Nginx 配置文件修改
|
|
|
|
### Task 5: 创建新的 Nginx 配置文件
|
|
|
|
**Files:**
|
|
- Modify: `conf/emotion-museum.conf`
|
|
|
|
- [ ] **Step 1: 备份当前配置文件**
|
|
|
|
```bash
|
|
cp conf/emotion-museum.conf conf/emotion-museum.conf.backup
|
|
```
|
|
|
|
- [ ] **Step 2: 修改 Nginx 配置为域名版本**
|
|
|
|
```nginx
|
|
# Emotion Museum - 域名部署配置
|
|
# 域名:lifescript.happylifeos.com
|
|
# 配置路径:/etc/nginx/sites-available/lifescript.happylifeos.com.conf
|
|
|
|
# HTTP 服务器 - 强制跳转 HTTPS
|
|
server {
|
|
listen 80;
|
|
server_name lifescript.happylifeos.com;
|
|
|
|
# 强制跳转 HTTPS
|
|
return 301 https://$server_name$request_uri;
|
|
}
|
|
|
|
# HTTPS 服务器
|
|
server {
|
|
listen 443 ssl http2;
|
|
server_name lifescript.happylifeos.com;
|
|
|
|
# SSL 证书配置
|
|
ssl_certificate /etc/letsencrypt/live/lifescript.happylifeos.com/fullchain.pem;
|
|
ssl_certificate_key /etc/letsencrypt/live/lifescript.happylifeos.com/privkey.pem;
|
|
|
|
# SSL 优化配置
|
|
ssl_protocols TLSv1.2 TLSv1.3;
|
|
ssl_prefer_server_ciphers on;
|
|
ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384;
|
|
ssl_session_cache shared:SSL:10m;
|
|
ssl_session_timeout 10m;
|
|
|
|
# HSTS (可选,生产环境建议开启)
|
|
# add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always;
|
|
|
|
# 根路径 - 用户前端应用
|
|
location / {
|
|
alias /data/www/emotion-museum/;
|
|
autoindex off;
|
|
|
|
# 处理 Vue Router 的 history 模式
|
|
try_files $uri $uri/ /index.html;
|
|
|
|
# HTML 文件不缓存
|
|
location ~ \.html?$ {
|
|
add_header Cache-Control "no-cache, no-store, must-revalidate";
|
|
add_header Pragma "no-cache";
|
|
add_header Expires "0";
|
|
}
|
|
|
|
# 静态资源缓存 1 年
|
|
location ~ \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$ {
|
|
add_header Cache-Control "public, max-age=31536000, immutable";
|
|
expires 1y;
|
|
}
|
|
}
|
|
|
|
# 管理后台应用路径
|
|
location /emotion-museum-admin/ {
|
|
alias /data/www/emotion-museum-admin/;
|
|
autoindex off;
|
|
|
|
# 处理 Vue Router 的 history 模式
|
|
try_files $uri $uri/ /emotion-museum-admin/index.html;
|
|
|
|
# HTML 文件不缓存
|
|
location ~ \.html?$ {
|
|
add_header Cache-Control "no-cache, no-store, must-revalidate";
|
|
add_header Pragma "no-cache";
|
|
add_header Expires "0";
|
|
}
|
|
|
|
# 静态资源缓存 1 年
|
|
location ~ \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$ {
|
|
add_header Cache-Control "public, max-age=31536000, immutable";
|
|
expires 1y;
|
|
}
|
|
}
|
|
|
|
# 处理不带末尾斜杠的 /emotion-museum-admin 请求
|
|
location = /emotion-museum-admin {
|
|
rewrite ^(.*)$ $1/ permanent;
|
|
}
|
|
|
|
# Life-Script 应用路径
|
|
location /life-script/ {
|
|
alias /data/www/life-script/;
|
|
autoindex off;
|
|
|
|
# 处理 React Router 的 history 模式
|
|
try_files $uri $uri/ /life-script/index.html;
|
|
|
|
# HTML 文件不缓存
|
|
location ~ \.html?$ {
|
|
add_header Cache-Control "no-cache, no-store, must-revalidate";
|
|
add_header Pragma "no-cache";
|
|
add_header Expires "0";
|
|
}
|
|
|
|
# 静态资源缓存 1 年
|
|
location ~ \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$ {
|
|
add_header Cache-Control "public, max-age=31536000, immutable";
|
|
expires 1y;
|
|
}
|
|
}
|
|
|
|
# 处理不带末尾斜杠的 /life-script 请求
|
|
location = /life-script {
|
|
rewrite ^ /life-script/ last;
|
|
}
|
|
|
|
# 后端 API 代理
|
|
location /api {
|
|
proxy_pass http://127.0.0.1:19089;
|
|
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;
|
|
}
|
|
|
|
# WebSocket 代理
|
|
location /ws {
|
|
proxy_pass http://127.0.0.1:19089;
|
|
proxy_http_version 1.1;
|
|
proxy_set_header Upgrade $http_upgrade;
|
|
proxy_set_header Connection "upgrade";
|
|
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;
|
|
|
|
# WebSocket 超时设置
|
|
proxy_connect_timeout 7d;
|
|
proxy_send_timeout 7d;
|
|
proxy_read_timeout 7d;
|
|
}
|
|
|
|
# 健康检查端点
|
|
location /health {
|
|
access_log off;
|
|
return 200 "healthy\n";
|
|
add_header Content-Type text/plain;
|
|
}
|
|
|
|
# 禁止访问敏感文件
|
|
location ~ /\. {
|
|
deny all;
|
|
access_log off;
|
|
log_not_found off;
|
|
}
|
|
|
|
access_log /var/log/nginx/lifescript_access.log;
|
|
error_log /var/log/nginx/lifescript_error.log;
|
|
}
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add conf/emotion-museum.conf
|
|
git commit -m "config: 更新 nginx 配置为域名访问,添加 HTTPS SSL 支持"
|
|
```
|
|
|
|
---
|
|
|
|
## Chunk 4: 部署脚本修改
|
|
|
|
### Task 6: 创建新域名下的一键部署脚本
|
|
|
|
**Files:**
|
|
- Create: `deploy-to-prod.sh`
|
|
|
|
- [ ] **Step 1: 创建部署脚本**
|
|
|
|
```bash
|
|
#!/bin/bash
|
|
|
|
# 情绪博物馆 - 域名部署脚本
|
|
# 域名:lifescript.happylifeos.com
|
|
# 使用方法:bash deploy-to-prod.sh [ssl|frontend|admin|life-script|backend|all]
|
|
|
|
set -e
|
|
|
|
# 颜色定义
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
BLUE='\033[0;34m'
|
|
NC='\033[0m'
|
|
|
|
# 日志函数
|
|
log_info() {
|
|
echo -e "${GREEN}[INFO]${NC} $1"
|
|
}
|
|
|
|
log_warn() {
|
|
echo -e "${YELLOW}[WARN]${NC} $1"
|
|
}
|
|
|
|
log_error() {
|
|
echo -e "${RED}[ERROR]${NC} $1"
|
|
}
|
|
|
|
log_section() {
|
|
echo ""
|
|
echo -e "${BLUE}╔════════════════════════════════════════════════════════════════╗${NC}"
|
|
echo -e "${BLUE}║${NC} $1"
|
|
echo -e "${BLUE}╚════════════════════════════════════════════════════════════════╝${NC}"
|
|
echo ""
|
|
}
|
|
|
|
# 服务器配置
|
|
SERVER_IP="101.200.208.45"
|
|
USERNAME="root"
|
|
DOMAIN="lifescript.happylifeos.com"
|
|
|
|
# SSL 证书申请
|
|
deploy_ssl() {
|
|
log_section "申请 SSL 证书"
|
|
|
|
if [ ! -f "tools/deploy-ssl-cert.py" ]; then
|
|
log_error "SSL 证书脚本不存在:tools/deploy-ssl-cert.py"
|
|
return 1
|
|
fi
|
|
|
|
log_info "在服务器上执行 SSL 证书申请..."
|
|
scp tools/deploy-ssl-cert.py ${USERNAME}@${SERVER_IP}:/tmp/deploy-ssl-cert.py
|
|
ssh ${USERNAME}@${SERVER_IP} "python3 /tmp/deploy-ssl-cert.py"
|
|
|
|
if [ $? -eq 0 ]; then
|
|
log_info "✅ SSL 证书申请完成"
|
|
return 0
|
|
else
|
|
log_error "❌ SSL 证书申请失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 部署后端
|
|
deploy_backend() {
|
|
log_section "部署后端服务"
|
|
|
|
if [ ! -f "backend-single/deploy.sh" ]; then
|
|
log_error "后端部署脚本不存在"
|
|
return 1
|
|
fi
|
|
|
|
log_info "执行后端部署..."
|
|
cd backend-single
|
|
bash deploy.sh remote
|
|
local result=$?
|
|
cd ..
|
|
|
|
if [ $result -eq 0 ]; then
|
|
log_info "✅ 后端部署完成"
|
|
return 0
|
|
else
|
|
log_error "❌ 后端部署失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 部署前端
|
|
deploy_frontend() {
|
|
log_section "部署用户前端"
|
|
|
|
if [ ! -f "web/deploy.sh" ]; then
|
|
log_error "前端部署脚本不存在"
|
|
return 1
|
|
fi
|
|
|
|
log_info "执行前端部署..."
|
|
cd web
|
|
bash deploy.sh
|
|
local result=$?
|
|
cd ..
|
|
|
|
if [ $result -eq 0 ]; then
|
|
log_info "✅ 前端部署完成"
|
|
return 0
|
|
else
|
|
log_error "❌ 前端部署失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 部署管理后台
|
|
deploy_admin() {
|
|
log_section "部署管理后台"
|
|
|
|
if [ ! -f "web-admin/deploy.sh" ]; then
|
|
log_error "管理后台部署脚本不存在"
|
|
return 1
|
|
fi
|
|
|
|
log_info "执行管理后台部署..."
|
|
cd web-admin
|
|
bash deploy.sh
|
|
local result=$?
|
|
cd ..
|
|
|
|
if [ $result -eq 0 ]; then
|
|
log_info "✅ 管理后台部署完成"
|
|
return 0
|
|
else
|
|
log_error "❌ 管理后台部署失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 部署 Life-Script
|
|
deploy_life_script() {
|
|
log_section "部署 Life-Script"
|
|
|
|
if [ ! -f "life-script/deploy.sh" ]; then
|
|
log_error "Life-Script 部署脚本不存在"
|
|
return 1
|
|
fi
|
|
|
|
log_info "执行 Life-Script 部署..."
|
|
cd life-script
|
|
bash deploy.sh
|
|
local result=$?
|
|
cd ..
|
|
|
|
if [ $result -eq 0 ]; then
|
|
log_info "✅ Life-Script 部署完成"
|
|
return 0
|
|
else
|
|
log_error "❌ Life-Script 部署失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 部署 Nginx 配置
|
|
deploy_nginx() {
|
|
log_section "部署 Nginx 配置"
|
|
|
|
if [ ! -f "conf/emotion-museum.conf" ]; then
|
|
log_error "Nginx 配置文件不存在"
|
|
return 1
|
|
fi
|
|
|
|
log_info "上传 Nginx 配置文件..."
|
|
scp conf/emotion-museum.conf ${USERNAME}@${SERVER_IP}:/etc/nginx/sites-available/${DOMAIN}.conf
|
|
|
|
log_info "启用站点配置..."
|
|
ssh ${USERNAME}@${SERVER_IP} "ln -snf /etc/nginx/sites-available/${DOMAIN}.conf /etc/nginx/sites-enabled/${DOMAIN}.conf"
|
|
|
|
log_info "移除默认配置(如果存在冲突)..."
|
|
ssh ${USERNAME}@${SERVER_IP} "rm -f /etc/nginx/sites-enabled/default"
|
|
|
|
log_info "验证 Nginx 配置..."
|
|
if ssh ${USERNAME}@${SERVER_IP} "nginx -t"; then
|
|
log_info "Nginx 配置验证通过"
|
|
ssh ${USERNAME}@${SERVER_IP} "systemctl reload nginx"
|
|
log_info "✅ Nginx 配置重载完成"
|
|
return 0
|
|
else
|
|
log_error "❌ Nginx 配置验证失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 主程序
|
|
main() {
|
|
DEPLOY_TYPE="${1:-all}"
|
|
|
|
log_section "情绪博物馆 - 域名部署"
|
|
log_info "域名:https://${DOMAIN}"
|
|
log_info "部署类型:${DEPLOY_TYPE}"
|
|
log_info "部署时间:$(date '+%Y-%m-%d %H:%M:%S')"
|
|
|
|
case "$DEPLOY_TYPE" in
|
|
ssl)
|
|
deploy_ssl
|
|
;;
|
|
backend)
|
|
deploy_backend
|
|
;;
|
|
frontend)
|
|
deploy_frontend
|
|
;;
|
|
admin)
|
|
deploy_admin
|
|
;;
|
|
life-script)
|
|
deploy_life_script
|
|
;;
|
|
nginx)
|
|
deploy_nginx
|
|
;;
|
|
all)
|
|
deploy_ssl
|
|
deploy_nginx
|
|
deploy_backend
|
|
deploy_frontend
|
|
deploy_admin
|
|
deploy_life_script
|
|
;;
|
|
*)
|
|
log_error "无效的部署类型:$DEPLOY_TYPE"
|
|
echo "使用方法:bash deploy-to-prod.sh [ssl|backend|frontend|admin|life-script|nginx|all]"
|
|
exit 1
|
|
;;
|
|
esac
|
|
|
|
log_section "部署完成"
|
|
|
|
if [ "$DEPLOY_TYPE" = "all" ] || [ "$DEPLOY_TYPE" = "ssl" ]; then
|
|
log_info "📋 验证 SSL 证书:运行 python3 tools/deploy-ssl-cert.py --verify"
|
|
fi
|
|
|
|
if [ "$DEPLOY_TYPE" = "all" ] || [ "$DEPLOY_TYPE" = "nginx" ]; then
|
|
log_info "🌐 访问地址:"
|
|
log_info " 用户前端:https://${DOMAIN}/"
|
|
log_info " 管理后台:https://${DOMAIN}/emotion-museum-admin/"
|
|
log_info " Life-Script: https://${DOMAIN}/life-script/"
|
|
log_info " API 地址:https://${DOMAIN}/api"
|
|
log_info " WebSocket: wss://${DOMAIN}/ws"
|
|
fi
|
|
}
|
|
|
|
main "$@"
|
|
```
|
|
|
|
- [ ] **Step 2: 添加执行权限**
|
|
|
|
```bash
|
|
chmod +x deploy-to-prod.sh
|
|
```
|
|
|
|
- [ ] **Step 3: 添加元信息**
|
|
|
|
在文件顶部添加:
|
|
```bash
|
|
#!/bin/bash
|
|
# Author: [创建人姓名]
|
|
# Created: 2026-03-17
|
|
# Purpose: 域名部署入口脚本 - 支持 SSL 证书申请、前端、后端、nginx 配置部署
|
|
```
|
|
|
|
- [ ] **Step 4: 提交**
|
|
|
|
```bash
|
|
git add deploy-to-prod.sh
|
|
git commit -m "feat: 添加域名一键部署脚本"
|
|
```
|
|
|
|
### Task 7: 修改用户前端部署脚本
|
|
|
|
**Files:**
|
|
- Modify: `web/deploy.sh`
|
|
|
|
- [ ] **Step 1: 读取当前文件**
|
|
|
|
- [ ] **Step 2: 修改远程路径和访问地址提示**
|
|
|
|
将访问地址提示修改为:
|
|
```bash
|
|
echo "✅ 部署完成!"
|
|
echo "📱 访问地址:https://lifescript.happylifeos.com/"
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add web/deploy.sh
|
|
git commit -m "config: 更新前端部署脚本访问地址提示"
|
|
```
|
|
|
|
### Task 8: 修改管理后台部署脚本
|
|
|
|
**Files:**
|
|
- Modify: `web-admin/deploy.sh`
|
|
|
|
- [ ] **Step 1: 读取当前文件**
|
|
|
|
- [ ] **Step 2: 修改访问地址提示**
|
|
|
|
将访问地址提示修改为:
|
|
```bash
|
|
echo "✅ 管理后台部署完成!"
|
|
echo "🔧 访问地址:https://lifescript.happylifeos.com/emotion-museum-admin/"
|
|
```
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add web-admin/deploy.sh
|
|
git commit -m "config: 更新管理后台部署脚本访问地址提示"
|
|
```
|
|
|
|
---
|
|
|
|
## Chunk 5: 后端配置与验证
|
|
|
|
### Task 9: 检查并修改后端 CORS 配置
|
|
|
|
**Files:**
|
|
- Modify: `backend-single/src/main/resources/application.yml` (或相应配置文件)
|
|
|
|
- [ ] **Step 1: 读取当前后端配置文件**
|
|
|
|
检查 CORS 配置是否允许新域名访问
|
|
|
|
- [ ] **Step 2: 如需修改,添加域名白名单**
|
|
|
|
确保 CORS 配置包含 `https://lifescript.happylifeos.com`
|
|
|
|
- [ ] **Step 3: 提交**
|
|
|
|
```bash
|
|
git add backend-single/src/main/resources/application.yml
|
|
git commit -m "config: 更新 CORS 配置允许新域名访问"
|
|
```
|
|
|
|
### Task 10: 执行部署验证
|
|
|
|
**Files:**
|
|
- 无需修改文件
|
|
|
|
- [ ] **Step 1: 本地类型检查**
|
|
|
|
```bash
|
|
cd web && npm run type-check
|
|
cd ../web-admin && npm run type-check
|
|
cd ../life-script && npm run type-check
|
|
```
|
|
|
|
- [ ] **Step 2: 推送代码到服务器**
|
|
|
|
```bash
|
|
git push origin master
|
|
```
|
|
|
|
- [ ] **Step 3: 执行 SSL 证书申请**
|
|
|
|
```bash
|
|
bash deploy-to-prod.sh ssl
|
|
```
|
|
|
|
- [ ] **Step 4: 执行完整部署**
|
|
|
|
```bash
|
|
bash deploy-to-prod.sh all
|
|
```
|
|
|
|
- [ ] **Step 5: 浏览器验证**
|
|
|
|
使用浏览器访问以下地址验证:
|
|
- https://lifescript.happylifeos.com/
|
|
- https://lifescript.happylifeos.com/emotion-museum-admin/
|
|
- https://lifescript.happylifeos.com/life-script/
|
|
- https://lifescript.happylifeos.com/api/health
|
|
|
|
- [ ] **Step 6: 验证浏览器 Console 无错误**
|
|
|
|
确保所有页面:
|
|
- ✅ 正常加载
|
|
- ✅ 无 JavaScript 错误
|
|
- ✅ API 请求成功
|
|
- ✅ WebSocket 连接成功
|
|
|
|
---
|
|
|
|
## 部署总结
|
|
|
|
### 文件清单
|
|
|
|
| 文件 | 操作 | 说明 |
|
|
|------|------|------|
|
|
| `tools/deploy-ssl-cert.py` | 创建 | SSL 证书申请与自动续期脚本 |
|
|
| `web/.env.production` | 修改 | 前端 API 配置改为相对路径 |
|
|
| `web-admin/.env.production` | 修改 | 管理后台 API 配置改为相对路径 |
|
|
| `life-script/.env.production` | 修改 | Life-Script API 配置改为相对路径 |
|
|
| `conf/emotion-museum.conf` | 修改 | Nginx 域名配置 |
|
|
| `deploy-to-prod.sh` | 创建 | 域名部署入口脚本 |
|
|
| `web/deploy.sh` | 修改 | 更新访问地址提示 |
|
|
| `web-admin/deploy.sh` | 修改 | 更新访问地址提示 |
|
|
|
|
### 部署命令
|
|
|
|
```bash
|
|
# 1. 提交所有配置变更
|
|
git add .
|
|
git commit -m "chore: 域名部署配置更新"
|
|
git push
|
|
|
|
# 2. 申请 SSL 证书
|
|
bash deploy-to-prod.sh ssl
|
|
|
|
# 3. 部署所有服务
|
|
bash deploy-to-prod.sh all
|
|
```
|
|
|
|
### 验证检查清单
|
|
|
|
- [ ] SSL 证书有效且自动续期已配置
|
|
- [ ] HTTPS 强制跳转生效
|
|
- [ ] 用户前端正常访问
|
|
- [ ] 管理后台正常访问
|
|
- [ ] Life-Script 正常访问
|
|
- [ ] API 代理正常
|
|
- [ ] WebSocket 连接正常
|
|
- [ ] 浏览器 Console 无错误
|
|
- [ ] 所有静态资源正确加载
|