File size: 11,418 Bytes
2b8ef3a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
#!/bin/bash

# 设置颜色
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
RED='\033[0;31m'
NC='\033[0m' # No Color

# 日志函数
log() {
  echo -e "${GREEN}[$(date '+%Y-%m-%d %H:%M:%S')] $1${NC}"
}

log "数据库同步脚本启动"

# 检查环境变量
if [ -z "$WEBDAV_URL" ] || [ -z "$WEBDAV_USERNAME" ] || [ -z "$WEBDAV_PASSWORD" ]; then
  echo -e "${YELLOW}WebDAV 配置不完整,跳过同步${NC}"
  exit 0
fi

# 同步间隔(秒)
INTERVAL=${SYNC_INTERVAL:-600}
log "同步间隔设置为 $INTERVAL 秒"

# 数据库路径
DB_PATH=${DB_PATH:-/app/data/video-player.db}
log "数据库路径: $DB_PATH"

# 确保数据目录存在
mkdir -p $(dirname $DB_PATH)

# 使用 TMPDIR 环境变量
TEMP_DIR=${TMPDIR:-"/tmp"}
DB_DOWNLOAD_PATH="${TEMP_DIR}/video-player.db.download"
BACKUP_DIR="${TEMP_DIR}/backup"

# 设置备份路径
WEBDAV_BACKUP_PATH=${WEBDAV_BACKUP_PATH:-"video-player-data"}
FULL_WEBDAV_URL="${WEBDAV_URL}"

# 确保 WEBDAV_URL 以斜杠结尾
if [ "${WEBDAV_URL: -1}" != "/" ]; then
    WEBDAV_URL="${WEBDAV_URL}/"
fi

if [ -n "$WEBDAV_BACKUP_PATH" ]; then
    FULL_WEBDAV_URL="${WEBDAV_URL}${WEBDAV_BACKUP_PATH}"
    
    # 确保 FULL_WEBDAV_URL 以斜杠结尾
    if [ "${FULL_WEBDAV_URL: -1}" != "/" ]; then
        FULL_WEBDAV_URL="${FULL_WEBDAV_URL}/"
    fi
fi

log "WebDAV 同步已配置:"
log "  URL: ${FULL_WEBDAV_URL}"
log "  同步间隔: ${INTERVAL}秒"

# 检查 WebDAV 连接和权限
check_webdav_connection() {
    log "检查 WebDAV 连接和权限..."
    
    # 尝试列出根目录
    HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X PROPFIND -H "Depth: 1" "${FULL_WEBDAV_URL}/")
    
    if [ "$HTTP_CODE" = "207" ] || [ "$HTTP_CODE" = "200" ]; then
        log "WebDAV 连接成功"
        return 0
    elif [ "$HTTP_CODE" = "404" ]; then
        echo -e "${YELLOW}WebDAV 路径不存在,尝试创建...${NC}"
        
        # 尝试创建主目录
        create_main_dir
        return $?
    elif [ "$HTTP_CODE" = "401" ]; then
        echo -e "${RED}WebDAV 认证失败,请检查用户名和密码${NC}"
        return 1
    elif [ "$HTTP_CODE" = "403" ]; then
        echo -e "${RED}WebDAV 权限不足,无法访问${NC}"
        return 1
    else
        echo -e "${RED}WebDAV 连接失败: HTTP 状态码 ${HTTP_CODE}${NC}"
        return 1
    fi
}

# 创建主目录
create_main_dir() {
    # 如果路径包含多级目录,需要逐级创建
    IFS='/' read -ra DIRS <<< "$WEBDAV_BACKUP_PATH"
    CURRENT_PATH=""
    
    for DIR in "${DIRS[@]}"; do
        if [ -z "$DIR" ]; then
            continue
        fi
        
        if [ -z "$CURRENT_PATH" ]; then
            CURRENT_PATH="$DIR"
        else
            CURRENT_PATH="$CURRENT_PATH/$DIR"
        fi
        
        HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X MKCOL "${WEBDAV_URL}${CURRENT_PATH}/")
        
        if [ "$HTTP_CODE" = "201" ]; then
            log "创建目录成功: ${CURRENT_PATH}"
        elif [ "$HTTP_CODE" = "405" ]; then
            echo -e "${YELLOW}目录已存在: ${CURRENT_PATH}${NC}"
        elif [ "$HTTP_CODE" = "409" ]; then
            echo -e "${YELLOW}父目录不存在: ${CURRENT_PATH}${NC}"
            return 1
        elif [ "$HTTP_CODE" = "403" ]; then
            echo -e "${RED}无权限创建目录: ${CURRENT_PATH}${NC}"
            return 1
        else
            echo -e "${RED}创建目录失败: ${CURRENT_PATH}, HTTP 状态码 ${HTTP_CODE}${NC}"
            return 1
        fi
    done
    
    # 创建备份子目录
    HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X MKCOL "${FULL_WEBDAV_URL}backups/")
    
    if [ "$HTTP_CODE" = "201" ]; then
        log "创建备份目录成功"
        return 0
    elif [ "$HTTP_CODE" = "405" ]; then
        echo -e "${YELLOW}备份目录已存在${NC}"
        return 0
    elif [ "$HTTP_CODE" = "403" ]; then
        echo -e "${RED}无权限创建备份目录${NC}"
        return 1
    else
        echo -e "${RED}创建备份目录失败: HTTP 状态码 ${HTTP_CODE}${NC}"
        return 1
    fi
}

# 从 WebDAV 下载数据库
download_db() {
    log "从 WebDAV 下载数据库..."
    
    # 使用 curl 下载数据库文件
    HTTP_CODE=$(curl -s -o "${DB_DOWNLOAD_PATH}" -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" "${FULL_WEBDAV_URL}video-player.db")
    
    if [ "$HTTP_CODE" = "200" ]; then
        # 检查文件是否为有效的 SQLite 数据库
        if [ -s "${DB_DOWNLOAD_PATH}" ]; then
            log "成功下载数据库文件"
            
            # 如果当前没有数据库或下载的数据库更新,则使用下载的数据库
            if [ ! -f "$DB_PATH" ] || [ $(stat -c %Y "${DB_DOWNLOAD_PATH}" 2>/dev/null || echo 0) -gt $(stat -c %Y "$DB_PATH" 2>/dev/null || echo 0) ]; then
                cp "${DB_DOWNLOAD_PATH}" "$DB_PATH"
                log "已更新本地数据库"
            else
                echo -e "${YELLOW}本地数据库已是最新版本${NC}"
                rm "${DB_DOWNLOAD_PATH}"
            fi
        else
            echo -e "${RED}下载的文件不是有效的文件${NC}"
            rm "${DB_DOWNLOAD_PATH}"
        fi
    elif [ "$HTTP_CODE" = "404" ]; then
        echo -e "${YELLOW}WebDAV 上不存在数据库文件,将使用本地数据库${NC}"
        rm -f "${DB_DOWNLOAD_PATH}"
    else
        echo -e "${RED}下载数据库失败: HTTP 状态码 ${HTTP_CODE}${NC}"
        rm -f "${DB_DOWNLOAD_PATH}"
    fi
}

# 上传数据库到 WebDAV
upload_db() {
    log "上传数据库到 WebDAV..."
    
    # 确保数据库文件存在
    if [ ! -f "$DB_PATH" ]; then
        echo -e "${YELLOW}本地数据库文件不存在,跳过上传${NC}"
        return
    fi
    
    # 确保备份目录存在
    log "确保备份目录存在..."
    HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X PROPFIND -H "Depth: 1" "${FULL_WEBDAV_URL}backups/")
    
    if [ "$HTTP_CODE" != "207" ] && [ "$HTTP_CODE" != "200" ]; then
        log "备份目录不存在,尝试创建..."
        HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X MKCOL "${FULL_WEBDAV_URL}backups/")
        
        if [ "$HTTP_CODE" != "201" ] && [ "$HTTP_CODE" != "405" ]; then
            echo -e "${RED}创建备份目录失败: HTTP 状态码 ${HTTP_CODE}${NC}"
            echo -e "${YELLOW}将跳过备份上传,但会继续上传主数据库文件${NC}"
        else
            log "备份目录创建成功"
        fi
    else
        log "备份目录已存在"
    fi
    
    # 创建备份目录(如果不存在)
    mkdir -p "${BACKUP_DIR}"
    
    # 创建带时间戳的备份
    TIMESTAMP=$(date +%Y%m%d_%H%M%S)
    cp "$DB_PATH" "${BACKUP_DIR}/video-player_${TIMESTAMP}.db"
    
    # 上传当前数据库
    HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -T "$DB_PATH" "${FULL_WEBDAV_URL}video-player.db")
    
    if [ "$HTTP_CODE" = "201" ] || [ "$HTTP_CODE" = "204" ]; then
        log "成功上传数据库"
        
        # 尝试上传到备份目录
        HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -T "${BACKUP_DIR}/video-player_${TIMESTAMP}.db" "${FULL_WEBDAV_URL}backups/video-player_${TIMESTAMP}.db")
        
        if [ "$HTTP_CODE" = "201" ] || [ "$HTTP_CODE" = "204" ]; then
            log "成功上传数据库备份 video-player_${TIMESTAMP}.db"
        elif [ "$HTTP_CODE" = "403" ] || [ "$HTTP_CODE" = "405" ]; then
            log "备份目录上传失败,尝试直接上传到主目录..."
            # 尝试直接上传到主目录
            HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -T "${BACKUP_DIR}/video-player_${TIMESTAMP}.db" "${FULL_WEBDAV_URL}video-player_backup_${TIMESTAMP}.db")
            
            if [ "$HTTP_CODE" = "201" ] || [ "$HTTP_CODE" = "204" ]; then
                log "成功上传数据库备份到主目录 video-player_backup_${TIMESTAMP}.db"
            else
                echo -e "${YELLOW}上传数据库备份到主目录也失败: HTTP 状态码 ${HTTP_CODE}${NC}"
                echo -e "${YELLOW}备份文件路径: ${BACKUP_DIR}/video-player_${TIMESTAMP}.db${NC}"
                echo -e "${YELLOW}目标 URL: ${FULL_WEBDAV_URL}video-player_backup_${TIMESTAMP}.db${NC}"
            fi
        else
            echo -e "${YELLOW}上传数据库备份失败: HTTP 状态码 ${HTTP_CODE}${NC}"
            echo -e "${YELLOW}备份文件路径: ${BACKUP_DIR}/video-player_${TIMESTAMP}.db${NC}"
            echo -e "${YELLOW}目标 URL: ${FULL_WEBDAV_URL}backups/video-player_${TIMESTAMP}.db${NC}"
        fi
    else
        echo -e "${RED}上传数据库失败: HTTP 状态码 ${HTTP_CODE}${NC}"
    fi
    
    # 清理临时文件
    rm -rf "${BACKUP_DIR}"
}

# 清理旧备份
cleanup_old_backups() {
    log "清理旧备份..."
    
    # 首先尝试列出备份目录中的备份
    # 使用 curl 列出目录内容
    BACKUPS=$(curl -s -X PROPFIND -H "Depth: 1" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" "${FULL_WEBDAV_URL}backups/" | grep -o "backups/video-player_[0-9]\{8\}_[0-9]\{6\}.db" | sort)
    
    # 如果备份目录中没有备份,尝试列出主目录中的备份
    if [ -z "$BACKUPS" ]; then
        log "备份目录中没有找到备份,尝试在主目录中查找..."
        BACKUPS=$(curl -s -X PROPFIND -H "Depth: 1" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" "${FULL_WEBDAV_URL}" | grep -o "video-player_backup_[0-9]\{8\}_[0-9]\{6\}.db" | sort)
    fi
    
    # 计算要删除的备份数量(保留最新的5个)
    KEEP_COUNT=${BACKUP_KEEP_COUNT:-5}
    BACKUP_COUNT=$(echo "$BACKUPS" | wc -l)
    DELETE_COUNT=$((BACKUP_COUNT - KEEP_COUNT))
    
    if [ $DELETE_COUNT -gt 0 ]; then
        log "找到 ${BACKUP_COUNT} 个备份,将删除最旧的 ${DELETE_COUNT} 个"
        
        # 获取要删除的备份列表
        TO_DELETE=$(echo "$BACKUPS" | head -n $DELETE_COUNT)
        
        # 删除旧备份
        for BACKUP in $TO_DELETE; do
            HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" -u "$WEBDAV_USERNAME:$WEBDAV_PASSWORD" -X DELETE "${FULL_WEBDAV_URL}${BACKUP}")
            
            if [ "$HTTP_CODE" = "204" ] || [ "$HTTP_CODE" = "200" ]; then
                log "已删除旧备份: ${BACKUP}"
            else
                echo -e "${YELLOW}删除旧备份失败: ${BACKUP}, HTTP 状态码 ${HTTP_CODE}${NC}"
            fi
        done
    else
        log "备份数量不超过5个,无需清理"
    fi
}

# 首次启动时检查 WebDAV 连接并下载数据库
if check_webdav_connection; then
    download_db
else
    echo -e "${YELLOW}WebDAV 连接或权限问题,将使用本地数据库${NC}"
fi

# 主循环
while true; do
    log "开始同步过程: $(date)"
    
    # 检查 WebDAV 连接
    if check_webdav_connection; then
        # 上传数据库
        upload_db
        
        # 清理旧备份
        cleanup_old_backups
    else
        echo -e "${RED}WebDAV 连接或权限问题,跳过本次同步${NC}"
    fi
    
    log "同步完成,下次同步将在 ${INTERVAL} 秒后进行..."
    sleep $INTERVAL
done