增加rustdesk

This commit is contained in:
2026-04-24 11:13:49 +08:00
parent 18949f31ba
commit 3c5246afb5
17 changed files with 1780 additions and 19 deletions

28
rustdesk/.env.example Normal file
View File

@@ -0,0 +1,28 @@
# ===========================================
# RustDesk Server 配置文件
# ===========================================
# ----- 域名与证书 -----
# 访问域名(必填,如 rustdesk.example.com
RUSTDESK_DOMAIN=rustdesk.example.com
# Let's Encrypt 证书邮箱(必填)
CERTBOT_EMAIL=admin@example.com
# ----- Docker 镜像 -----
# RustDesk Server 镜像
RUSTDESK_IMAGE=rustdesk/rustdesk-server:latest
# ----- 网络 -----
# 绑定 IP默认 0.0.0.0 监听所有接口)
RUSTDESK_BIND_IP=0.0.0.0
# 仅允许加密连接1=是0=否)
ENCRYPTED_ONLY=1
# ----- 数据目录 -----
# RustDesk 数据目录(密钥对 + 数据库)
RUSTDESK_DATA_DIR=/var/lib/rustdesk
# 备份目录
BACKUP_DIR=/var/backups/rustdesk

150
rustdesk/README.md Normal file
View File

@@ -0,0 +1,150 @@
# RustDesk Server 部署指南
开源远程桌面服务器支持自托管数据完全自主可控。包含信号服务器hbbs和中继服务器hbbr客户端支持 Windows、macOS、Linux、iOS、Android。
## 功能特性
- 自建信号服务器hbbs+ 中继服务器hbbr
- 端到端加密,数据不经过第三方
- 支持 TCP 打洞和中继转发
- 自动生成 Ed25519 密钥对
- 支持仅加密连接模式
- 跨平台客户端支持
- WebSocket 支持(可通过 Nginx 代理)
## 技术栈
| 组件 | 版本 | 说明 |
|------|------|------|
| RustDesk Server | latest | 信号服务器 + 中继服务器 |
| Nginx | 系统包 | 反向代理 + HTTPSWebSocket |
| Docker | 最新版 | 容器运行环境 |
## 端口说明
| 端口 | 协议 | 服务 | 说明 |
|------|------|------|------|
| 21115 | TCP | hbbs | NAT 类型测试 |
| 21116 | TCP | hbbs | TCP 打洞 / 连接请求 |
| 21116 | UDP | hbbs | ID 注册 / 心跳 |
| 21117 | TCP | hbbr | 中继流量转发 |
| 21118 | WS | hbbs | WebSocket仅本地监听 |
| 21119 | WS | hbbr | WebSocket仅本地监听 |
> **21115、21116、21117 需要从外网可访问**21118/21119 仅本地通过 Nginx 代理。
## 前置条件
1. 一台 Linux 服务器Ubuntu 22.04/24.04 推荐)
2. 一个已解析到服务器的域名(如 `rustdesk.example.com`
3. 服务器 80/443 端口可从外网访问
4. 服务器 21115-21117 端口可从外网访问
## 目录结构
```
rustdesk/
├── docker-compose.yml # 容器编排hbbs + hbbr
├── .env.example # 配置模板
├── deploy.sh # 一键部署脚本
├── backup.sh # 备份脚本
├── uninstall.sh # 完全卸载脚本
├── nginx/
│ └── rustdesk.conf # Nginx 反向代理配置
└── README.md # 本文件
```
服务器上的数据目录:
```
/var/lib/rustdesk/ # RustDesk 数据(密钥对 + 数据库)
/var/backups/rustdesk/ # 备份文件
```
## 快速部署
### 第一步:上传文件到服务器
```bash
# 在本地执行,上传 rustdesk 目录
scp -r rustdesk/ root@<服务器IP>:/opt/rustdesk
# 如果服务器上还没有部署过 base首台服务或全新服务器还需上传 base
scp -r base/ root@<服务器IP>:/opt/base
```
### 第二步:登录服务器执行部署
```bash
ssh root@<服务器IP>
# 如果是全新服务器,先安装基础环境
cd /opt/base
cp .env.example .env
bash setup.sh
# 部署 RustDesk
cd /opt/rustdesk
bash deploy.sh
# 首次运行会生成 .env按提示修改配置后重新运行
vi .env
bash deploy.sh
```
### 第三步:配置域名解析
在域名服务商(如阿里云 DNS添加 A 记录:
| 记录类型 | 主机记录 | 记录值 |
|----------|----------|--------|
| A | rustdesk | `<服务器公网IP>` |
### 第四步:配置 RustDesk 客户端
1. 下载并安装 [RustDesk 客户端](https://rustdesk.com/zh/)
2. 打开客户端,进入 **设置 → 网络 → ID/中继服务器**
3. 填入以下信息:
| 配置项 | 值 |
|--------|-----|
| ID 服务器 | `rustdesk.yourdomain.com` |
| 中继服务器 | `rustdesk.yourdomain.com` |
| Key | 部署完成时显示的公钥 |
> **公钥获取方法:** 部署脚本完成时会自动显示公钥,也可手动查看:
>
> ```bash
> cat /var/lib/rustdesk/id_ed25519.pub
> ```
## 常用操作
```bash
# 查看服务状态
cd /opt/rustdesk && docker compose ps
# 查看日志
cd /opt/rustdesk && docker compose logs -f
# 重启服务
cd /opt/rustdesk && docker compose restart
# 停止服务
cd /opt/rustdesk && docker compose down
# 备份数据
cd /opt/rustdesk && bash backup.sh
# 查看公钥
cat /var/lib/rustdesk/id_ed25519.pub
# 更新镜像
cd /opt/rustdesk && docker compose pull && docker compose up -d
```
## 安全建议
- 默认开启 `ENCRYPTED_ONLY=1`,仅允许加密连接
- 密钥对文件(`id_ed25519` / `id_ed25519.pub`)是最重要的数据,务必备份
- 如需更换密钥对,删除数据目录中的密钥文件后重启服务即可重新生成
- 建议定期执行 `bash backup.sh` 并将备份异地存储

69
rustdesk/backup.sh Normal file
View File

@@ -0,0 +1,69 @@
#!/usr/bin/env bash
set -euo pipefail
# ============================================
# RustDesk 备份脚本
# 备份数据目录(密钥对 + 数据库)+ 配置文件
# ============================================
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
cd "$SCRIPT_DIR"
# 加载配置
if [ -f .env ]; then
sed -i 's/\r$//' .env
set -a; source .env; set +a
else
echo "[ERROR] .env 文件不存在,请先运行 deploy.sh" >&2
exit 1
fi
# ===== 配置 =====
DATA_DIR="${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}"
BACKUP_BASE="${BACKUP_DIR:-/var/backups/rustdesk}"
TIMESTAMP=$(date +%Y%m%d_%H%M%S)
BACKUP_DIR_FULL="${BACKUP_BASE}/${TIMESTAMP}"
RETENTION_DAYS=30
echo "========== RustDesk 备份 =========="
echo "时间: $(date '+%Y-%m-%d %H:%M:%S')"
echo "数据目录: $DATA_DIR"
echo "备份目录: $BACKUP_DIR_FULL"
echo ""
mkdir -p "$BACKUP_DIR_FULL"
# ===== 备份数据目录(密钥对 + db=====
echo "[1/3] 备份 RustDesk 数据..."
if [ -d "$DATA_DIR" ]; then
tar czf "$BACKUP_DIR_FULL/rustdesk-data.tar.gz" -C "$(dirname "$DATA_DIR")" "$(basename "$DATA_DIR")"
echo " ✓ 数据目录已备份(含密钥对和数据库)"
else
echo " ⚠ 数据目录不存在: $DATA_DIR"
fi
# ===== 备份 docker-compose 和配置 =====
echo "[2/3] 备份部署配置..."
tar czf "$BACKUP_DIR_FULL/rustdesk-config.tar.gz" \
-C "$SCRIPT_DIR" \
docker-compose.yml .env nginx/ 2>/dev/null || true
echo " ✓ 配置已备份"
# ===== 清理旧备份 =====
echo "[3/3] 清理 ${RETENTION_DAYS} 天前的旧备份..."
find "$BACKUP_BASE" -maxdepth 1 -type d -mtime +${RETENTION_DAYS} -exec rm -rf {} \; 2>/dev/null || true
echo " ✓ 旧备份已清理"
# ===== 汇总 =====
echo ""
echo "========== 备份完成 =========="
TOTAL_SIZE=$(du -sh "$BACKUP_DIR_FULL" | cut -f1)
echo "备份位置: $BACKUP_DIR_FULL"
echo "备份大小: $TOTAL_SIZE"
echo ""
echo "备份内容:"
ls -lh "$BACKUP_DIR_FULL/"
echo ""
echo "恢复方法:"
echo " tar xzf $BACKUP_DIR_FULL/rustdesk-data.tar.gz -C $(dirname "$DATA_DIR")"
echo " cd $SCRIPT_DIR && docker compose restart"

233
rustdesk/deploy.sh Normal file
View File

@@ -0,0 +1,233 @@
#!/usr/bin/env bash
set -euo pipefail
# ============================================
# RustDesk Server 一键部署脚本
# 自动安装 Docker + Nginx + SSL + RustDesk
# ============================================
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
cd "$SCRIPT_DIR"
# ===== 加载公共基础函数 =====
BASE_DIR="$(cd "$SCRIPT_DIR/../base" 2>/dev/null && pwd)" || true
if [ -z "$BASE_DIR" ] || [ ! -f "$BASE_DIR/setup.sh" ]; then
echo "[ERROR] base/setup.sh 未找到" >&2
echo "请确保目录结构如下:" >&2
echo " /opt/base/setup.sh" >&2
echo " /opt/rustdesk/deploy.sh (当前脚本)" >&2
exit 1
fi
source "$BASE_DIR/setup.sh"
# =============================================================
# RustDesk 专用函数
# =============================================================
init_env() {
step "初始化 RustDesk 配置"
if [ ! -f .env ]; then
if [ ! -f .env.example ]; then
error "缺少 .env.example 模板文件"
exit 1
fi
cp .env.example .env
log "已生成 .env 文件"
echo ""
warn "┌─────────────────────────────────────────────────┐"
warn "│ 请编辑 .env 文件,至少修改以下配置: │"
warn "│ │"
warn "│ RUSTDESK_DOMAIN=rustdesk.yourdomain.com │"
warn "│ CERTBOT_EMAIL=you@yourdomain.com │"
warn "│ │"
warn "│ 编辑命令: vi $SCRIPT_DIR/.env │"
warn "│ 编辑完成后重新运行: bash deploy.sh │"
warn "└─────────────────────────────────────────────────┘"
exit 0
fi
fix_crlf .env
set -a; source .env; set +a
local has_error=0
if [[ -z "${RUSTDESK_DOMAIN:-}" ]] || [[ "${RUSTDESK_DOMAIN}" == "rustdesk.example.com" ]]; then
error "请在 .env 中将 RUSTDESK_DOMAIN 修改为你的实际域名"
has_error=1
fi
if [[ -z "${CERTBOT_EMAIL:-}" ]] || [[ "${CERTBOT_EMAIL}" == "admin@example.com" ]]; then
error "请在 .env 中将 CERTBOT_EMAIL 修改为你的实际邮箱"
has_error=1
fi
[ "$has_error" -eq 1 ] && { error "请修改 .env 后重新运行"; exit 1; }
log "配置检查通过"
log " 域名: ${RUSTDESK_DOMAIN}"
log " 邮箱: ${CERTBOT_EMAIL}"
}
create_dirs() {
step "创建数据目录"
local data_dir="${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}"
local backup_dir="${BACKUP_DIR:-/var/backups/rustdesk}"
mkdir -p "$data_dir" "$backup_dir"
log "数据目录: $data_dir"
log "备份目录: $backup_dir"
}
setup_firewall_rustdesk() {
step "配置 RustDesk 防火墙规则"
if command -v ufw &>/dev/null; then
ufw allow 21115/tcp comment "RustDesk NAT type test"
ufw allow 21116/tcp comment "RustDesk TCP hole punching"
ufw allow 21116/udp comment "RustDesk ID registration/heartbeat"
ufw allow 21117/tcp comment "RustDesk Relay"
log "已添加 UFW 规则 (21115-21117)"
elif command -v firewall-cmd &>/dev/null; then
firewall-cmd --permanent --add-port=21115/tcp
firewall-cmd --permanent --add-port=21116/tcp
firewall-cmd --permanent --add-port=21116/udp
firewall-cmd --permanent --add-port=21117/tcp
firewall-cmd --reload
log "已添加 firewalld 规则 (21115-21117)"
else
warn "未检测到防火墙管理工具,请手动放行端口 21115-21117"
fi
}
start_services() {
step "启动 RustDesk 服务"
log "正在拉取镜像..."
docker compose pull
log "正在启动容器..."
docker compose up -d
log "等待 RustDesk 就绪..."
sleep 5
# 检查容器是否正常运行
if docker compose ps | grep -q "running"; then
log "RustDesk 服务启动成功!"
else
warn "RustDesk 可能仍在启动中,请稍后检查: docker compose logs -f"
fi
}
show_public_key() {
step "获取公钥信息"
local data_dir="${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}"
local key_file="$data_dir/id_ed25519.pub"
if [ -f "$key_file" ]; then
local pub_key
pub_key=$(cat "$key_file")
echo ""
echo -e "${CYAN}══════════════════════════════════════════════════════════${NC}"
echo -e "${CYAN} RustDesk 公钥(客户端配置时需要):${NC}"
echo -e "${CYAN}══════════════════════════════════════════════════════════${NC}"
echo ""
echo -e " ${GREEN}${pub_key}${NC}"
echo ""
echo -e "${CYAN}══════════════════════════════════════════════════════════${NC}"
echo ""
else
warn "公钥文件尚未生成,服务启动后会自动创建"
warn "稍后可查看: cat $key_file"
fi
}
show_info() {
set -a; source .env; set +a
local server_ip
server_ip=$(curl -sf https://api.ipify.org 2>/dev/null || curl -sf https://ifconfig.me 2>/dev/null || echo "<服务器IP>")
echo ""
echo -e "${GREEN}╔══════════════════════════════════════════════════════════╗${NC}"
echo -e "${GREEN}║ RustDesk Server 部署完成! ║${NC}"
echo -e "${GREEN}╠══════════════════════════════════════════════════════════╣${NC}"
echo -e "${GREEN}${NC}"
echo -e "${GREEN}${NC} 服务器 IP: ${CYAN}${server_ip}${NC}"
echo -e "${GREEN}${NC} 域名: ${CYAN}${RUSTDESK_DOMAIN}${NC}"
echo -e "${GREEN}${NC}"
echo -e "${GREEN}${NC} hbbs 端口: 21115/tcp, 21116/tcp+udp"
echo -e "${GREEN}${NC} hbbr 端口: 21117/tcp"
echo -e "${GREEN}${NC}"
echo -e "${GREEN}${NC} 数据目录: ${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}"
echo -e "${GREEN}${NC} 备份目录: ${BACKUP_DIR:-/var/backups/rustdesk}"
echo -e "${GREEN}${NC}"
echo -e "${GREEN}╚══════════════════════════════════════════════════════════╝${NC}"
echo ""
show_public_key
echo "客户端配置:"
echo " ID 服务器: ${RUSTDESK_DOMAIN}"
echo " 中继服务器: ${RUSTDESK_DOMAIN}"
echo " Key: (上方公钥)"
echo ""
echo "常用命令:"
echo " 查看日志: cd $SCRIPT_DIR && docker compose logs -f"
echo " 重启服务: cd $SCRIPT_DIR && docker compose restart"
echo " 停止服务: cd $SCRIPT_DIR && docker compose down"
echo " 查看公钥: cat ${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}/id_ed25519.pub"
echo " 备份数据: cd $SCRIPT_DIR && bash backup.sh"
echo ""
}
# =============================================================
# 主流程
# =============================================================
main() {
echo -e "${CYAN}"
echo " ____ _ ____ _ "
echo " | _ \\ _ _ ___| |_| _ \\ ___ ___| | __"
echo " | |_) | | | / __| __| | | |/ _ \\/ __| |/ /"
echo " | _ <| |_| \\__ \\ |_| |_| | __/\\__ \\ < "
echo " |_| \\_\\\\__,_|___/\\__|____/ \\___||___/_|\\_\\ Deploy Script"
echo -e "${NC}"
echo ""
check_root
load_base_env "$BASE_DIR"
# Step 1: 系统初始化
init_system
# Step 2: 安装 Docker
install_docker
# Step 3: 安装 Nginx
install_nginx
# Step 4: 初始化配置
init_env
# Step 5: 配置 Docker 镜像加速
configure_docker_mirrors
# Step 6: 创建数据目录
create_dirs
# Step 7: 配置防火墙(基础 + RustDesk 端口)
setup_firewall_base
setup_firewall_rustdesk
# Step 8: 配置 SSL 证书
setup_ssl_cert "${RUSTDESK_DOMAIN}" "${CERTBOT_EMAIL}" "rustdesk"
# Step 9: 部署 Nginx 反向代理
deploy_nginx_conf "$SCRIPT_DIR/nginx/rustdesk.conf" "${RUSTDESK_DOMAIN}" "rustdesk"
# Step 10: 启动服务
start_services
# 显示部署信息
show_info
log "===== RustDesk Server 部署完成 ====="
}
main "$@"

View File

@@ -0,0 +1,44 @@
services:
rustdesk-hbbs:
image: ${RUSTDESK_IMAGE:-rustdesk/rustdesk-server:latest}
container_name: rustdesk-hbbs
restart: unless-stopped
command: hbbs
environment:
- TZ=Asia/Shanghai
- ENCRYPTED_ONLY=${ENCRYPTED_ONLY:-1}
volumes:
- ${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}:/root
- /etc/localtime:/etc/localtime:ro
ports:
- "${RUSTDESK_BIND_IP:-0.0.0.0}:21115:21115"
- "${RUSTDESK_BIND_IP:-0.0.0.0}:21116:21116"
- "${RUSTDESK_BIND_IP:-0.0.0.0}:21116:21116/udp"
- "127.0.0.1:21118:21118"
depends_on:
rustdesk-hbbr:
condition: service_started
logging:
driver: json-file
options:
max-size: "10m"
max-file: "3"
rustdesk-hbbr:
image: ${RUSTDESK_IMAGE:-rustdesk/rustdesk-server:latest}
container_name: rustdesk-hbbr
restart: unless-stopped
command: hbbr
environment:
- TZ=Asia/Shanghai
volumes:
- ${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}:/root
- /etc/localtime:/etc/localtime:ro
ports:
- "${RUSTDESK_BIND_IP:-0.0.0.0}:21117:21117"
- "127.0.0.1:21119:21119"
logging:
driver: json-file
options:
max-size: "10m"
max-file: "3"

View File

@@ -0,0 +1,76 @@
server {
listen 80;
listen [::]:80;
server_name __DOMAIN__;
location /.well-known/acme-challenge/ {
root /var/www/certbot;
}
location / {
return 301 https://$host$request_uri;
}
}
server {
listen 443 ssl;
listen [::]:443 ssl;
http2 on;
server_name __DOMAIN__;
# SSL 证书
ssl_certificate /etc/letsencrypt/live/__DOMAIN__/fullchain.pem;
ssl_certificate_key /etc/letsencrypt/live/__DOMAIN__/privkey.pem;
# SSL 参数
ssl_protocols TLSv1.2 TLSv1.3;
ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384;
ssl_prefer_server_ciphers off;
# 安全头
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 Strict-Transport-Security "max-age=63072000; includeSubDomains" always;
add_header Referrer-Policy strict-origin-when-cross-origin always;
# hbbs WebSocket (21118)
location /ws {
proxy_pass http://127.0.0.1:21118;
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_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_connect_timeout 60s;
proxy_send_timeout 300s;
proxy_read_timeout 300s;
}
# hbbr WebSocket (21119)
location /relay {
proxy_pass http://127.0.0.1:21119;
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_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_connect_timeout 60s;
proxy_send_timeout 300s;
proxy_read_timeout 300s;
}
# 默认页面
location / {
default_type text/html;
return 200 '<html><body><h1>RustDesk Server</h1><p>Server is running.</p></body></html>';
}
}

168
rustdesk/uninstall.sh Normal file
View File

@@ -0,0 +1,168 @@
#!/usr/bin/env bash
set -euo pipefail
# ============================================
# RustDesk 卸载脚本
# 停止容器 → 备份数据 → 清理容器/镜像/配置/数据
# ============================================
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
cd "$SCRIPT_DIR"
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
CYAN='\033[0;36m'
NC='\033[0m'
log() { echo -e "${GREEN}[INFO]${NC} $*"; }
warn() { echo -e "${YELLOW}[WARN]${NC} $*"; }
error() { echo -e "${RED}[ERROR]${NC} $*" >&2; }
# ===== 检查 root =====
if [ "$(id -u)" -ne 0 ]; then
error "请使用 root 用户运行: sudo bash uninstall.sh"
exit 1
fi
# ===== 加载配置 =====
if [ -f .env ]; then
sed -i 's/\r$//' .env
set -a
source .env
set +a
fi
RUSTDESK_DATA_DIR="${RUSTDESK_DATA_DIR:-/var/lib/rustdesk}"
BACKUP_DIR="${BACKUP_DIR:-/var/backups/rustdesk}"
RUSTDESK_DOMAIN="${RUSTDESK_DOMAIN:-}"
# ===== 确认操作 =====
echo ""
echo -e "${RED}╔══════════════════════════════════════════════════════╗${NC}"
echo -e "${RED}║ ⚠ 即将卸载 RustDesk 及所有数据 ⚠ ║${NC}"
echo -e "${RED}╚══════════════════════════════════════════════════════╝${NC}"
echo ""
echo "将执行以下操作:"
echo " 1. 停止并删除 RustDesk 容器hbbs + hbbr"
echo " 2. 删除 Docker 镜像"
echo " 3. 删除 Nginx 站点配置"
echo " 4. 删除 SSL 证书"
echo " 5. 删除 Certbot 自动续期定时任务"
echo " 6. 关闭防火墙端口 (21115-21117)"
echo ""
echo "涉及的数据目录:"
echo " RustDesk 数据: ${RUSTDESK_DATA_DIR}"
echo " 备份目录: ${BACKUP_DIR}"
echo " 部署目录: ${SCRIPT_DIR}"
echo ""
echo -e "${YELLOW}备份目录将保留,不会被删除。${NC}"
echo ""
read -r -p "确定要继续卸载吗?输入 YES 确认: " confirm
if [ "$confirm" != "YES" ]; then
log "已取消卸载"
exit 0
fi
# ===== 卸载前备份 =====
echo ""
read -r -p "是否在卸载前执行一次备份?(y/N): " do_backup
if [[ "$do_backup" =~ ^[Yy]$ ]]; then
if [ -f backup.sh ]; then
log "正在执行备份..."
bash backup.sh
log "备份完成"
else
warn "backup.sh 不存在,跳过备份"
fi
fi
# ===== 1. 停止并删除容器 =====
echo ""
log "正在停止并删除容器..."
if docker compose ps --quiet 2>/dev/null | grep -q .; then
docker compose down -v
log "容器已停止并删除"
else
log "没有运行中的容器"
fi
# ===== 2. 删除 Docker 镜像 =====
log "正在删除 Docker 镜像..."
RUSTDESK_IMAGE="${RUSTDESK_IMAGE:-rustdesk/rustdesk-server:latest}"
docker image rm "$RUSTDESK_IMAGE" 2>/dev/null && log "已删除镜像: $RUSTDESK_IMAGE" || true
# ===== 3. 删除 Nginx 配置 =====
log "正在清理 Nginx 配置..."
rm -f /etc/nginx/sites-enabled/rustdesk
rm -f /etc/nginx/sites-available/rustdesk
if command -v nginx &>/dev/null && nginx -t 2>/dev/null; then
systemctl reload nginx 2>/dev/null || true
log "Nginx 已重载"
fi
# ===== 4. 删除 SSL 证书 =====
if [ -n "$RUSTDESK_DOMAIN" ] && [ -d "/etc/letsencrypt/live/${RUSTDESK_DOMAIN}" ]; then
log "正在删除 SSL 证书: ${RUSTDESK_DOMAIN}..."
certbot delete --cert-name "${RUSTDESK_DOMAIN}" --non-interactive 2>/dev/null || true
log "SSL 证书已删除"
fi
# ===== 5. 清理 Certbot 定时任务 =====
remaining_certs=$(certbot certificates 2>/dev/null | grep -c "Certificate Name" || true)
if [ "$remaining_certs" -eq 0 ]; then
crontab -l 2>/dev/null | grep -v "certbot renew" | crontab - 2>/dev/null || true
log "已移除 Certbot 自动续期定时任务(无剩余证书)"
else
log "保留 Certbot 定时任务(还有 ${remaining_certs} 个其他证书)"
fi
# ===== 6. 关闭防火墙端口 =====
log "正在清理防火墙规则..."
if command -v ufw &>/dev/null; then
ufw delete allow 21115/tcp 2>/dev/null || true
ufw delete allow 21116/tcp 2>/dev/null || true
ufw delete allow 21116/udp 2>/dev/null || true
ufw delete allow 21117/tcp 2>/dev/null || true
log "已移除 UFW 规则"
elif command -v firewall-cmd &>/dev/null; then
firewall-cmd --permanent --remove-port=21115/tcp 2>/dev/null || true
firewall-cmd --permanent --remove-port=21116/tcp 2>/dev/null || true
firewall-cmd --permanent --remove-port=21116/udp 2>/dev/null || true
firewall-cmd --permanent --remove-port=21117/tcp 2>/dev/null || true
firewall-cmd --reload 2>/dev/null || true
log "已移除 firewalld 规则"
fi
# ===== 7. 删除数据目录 =====
echo ""
echo -e "${RED}以下目录将被永久删除:${NC}"
echo " ${RUSTDESK_DATA_DIR}"
echo ""
echo -e "${YELLOW}备份目录 ${BACKUP_DIR} 不会被删除。${NC}"
echo ""
read -r -p "确认删除数据目录?输入 DELETE 确认: " confirm_delete
if [ "$confirm_delete" = "DELETE" ]; then
rm -rf "$RUSTDESK_DATA_DIR"
log "已删除: ${RUSTDESK_DATA_DIR}"
else
warn "跳过数据目录删除"
fi
# ===== 8. 删除部署目录 =====
echo ""
read -r -p "是否删除部署目录 ${SCRIPT_DIR}(y/N): " del_deploy
if [[ "$del_deploy" =~ ^[Yy]$ ]]; then
cd /opt
rm -rf "$SCRIPT_DIR"
log "已删除部署目录"
else
warn "保留部署目录: ${SCRIPT_DIR}"
fi
# ===== 完成 =====
echo ""
log "RustDesk 卸载完成"
echo ""
echo "保留的内容:"
echo " 备份目录: ${BACKUP_DIR}"