mirror of
https://github.com/Awuqing/BackupX.git
synced 2026-05-06 20:02:41 +08:00
1. 审计日志:所有 handler 的 recordAudit 调用补充有意义的 detail, 包括创建/更新时记录类型、删除时记录 ID、设置变更时记录修改的 key 2. 版本号:Makefile 的 run/build 都通过 ldflags 注入 git 版本号, 开发模式不再显示 "dev"
193 lines
4.9 KiB
Go
193 lines
4.9 KiB
Go
package http
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"backupx/server/internal/apperror"
|
|
"backupx/server/internal/backup"
|
|
"backupx/server/internal/service"
|
|
"backupx/server/pkg/response"
|
|
"github.com/gin-gonic/gin"
|
|
)
|
|
|
|
type BackupRecordHandler struct {
|
|
service *service.BackupRecordService
|
|
auditService *service.AuditService
|
|
}
|
|
|
|
func NewBackupRecordHandler(recordService *service.BackupRecordService, auditService *service.AuditService) *BackupRecordHandler {
|
|
return &BackupRecordHandler{service: recordService, auditService: auditService}
|
|
}
|
|
|
|
func (h *BackupRecordHandler) List(c *gin.Context) {
|
|
filter, err := buildRecordFilter(c)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
items, err := h.service.List(c.Request.Context(), filter)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
response.Success(c, items)
|
|
}
|
|
|
|
func (h *BackupRecordHandler) Get(c *gin.Context) {
|
|
id, ok := parseUintParam(c, "id")
|
|
if !ok {
|
|
return
|
|
}
|
|
item, err := h.service.Get(c.Request.Context(), id)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
response.Success(c, item)
|
|
}
|
|
|
|
func (h *BackupRecordHandler) StreamLogs(c *gin.Context) {
|
|
id, ok := parseUintParam(c, "id")
|
|
if !ok {
|
|
return
|
|
}
|
|
detail, err := h.service.Get(c.Request.Context(), id)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
events := detail.LogEvents
|
|
completed := detail.Status != "running"
|
|
channel, cancel, err := h.service.SubscribeLogs(c.Request.Context(), id, 64)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
defer cancel()
|
|
c.Writer.Header().Set("Content-Type", "text/event-stream")
|
|
c.Writer.Header().Set("Cache-Control", "no-cache")
|
|
c.Writer.Header().Set("Connection", "keep-alive")
|
|
flusher, ok := c.Writer.(interface{ Flush() })
|
|
if !ok {
|
|
response.Error(c, apperror.Internal("BACKUP_RECORD_STREAM_UNSUPPORTED", "当前连接不支持日志流", nil))
|
|
return
|
|
}
|
|
for _, event := range events {
|
|
if err := writeSSEEvent(c.Writer, event); err != nil {
|
|
return
|
|
}
|
|
flusher.Flush()
|
|
}
|
|
if completed {
|
|
return
|
|
}
|
|
for {
|
|
select {
|
|
case <-c.Request.Context().Done():
|
|
return
|
|
case event, ok := <-channel:
|
|
if !ok {
|
|
return
|
|
}
|
|
if err := writeSSEEvent(c.Writer, event); err != nil {
|
|
return
|
|
}
|
|
flusher.Flush()
|
|
if event.Completed {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (h *BackupRecordHandler) Download(c *gin.Context) {
|
|
id, ok := parseUintParam(c, "id")
|
|
if !ok {
|
|
return
|
|
}
|
|
result, err := h.service.Download(c.Request.Context(), id)
|
|
if err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
defer result.Reader.Close()
|
|
c.Header("Content-Disposition", fmt.Sprintf("attachment; filename=%q", result.FileName))
|
|
c.Header("Content-Type", "application/octet-stream")
|
|
_, _ = io.Copy(c.Writer, result.Reader)
|
|
}
|
|
|
|
func (h *BackupRecordHandler) Restore(c *gin.Context) {
|
|
id, ok := parseUintParam(c, "id")
|
|
if !ok {
|
|
return
|
|
}
|
|
if err := h.service.Restore(c.Request.Context(), id); err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
recordAudit(c, h.auditService, "backup_record", "restore", "backup_record", fmt.Sprintf("%d", id), "", fmt.Sprintf("恢复备份记录 #%d", id))
|
|
response.Success(c, gin.H{"restored": true})
|
|
}
|
|
|
|
func (h *BackupRecordHandler) Delete(c *gin.Context) {
|
|
id, ok := parseUintParam(c, "id")
|
|
if !ok {
|
|
return
|
|
}
|
|
if err := h.service.Delete(c.Request.Context(), id); err != nil {
|
|
response.Error(c, err)
|
|
return
|
|
}
|
|
recordAudit(c, h.auditService, "backup_record", "delete", "backup_record", fmt.Sprintf("%d", id), "", fmt.Sprintf("删除备份记录 #%d", id))
|
|
response.Success(c, gin.H{"deleted": true})
|
|
}
|
|
|
|
func buildRecordFilter(c *gin.Context) (service.BackupRecordListInput, error) {
|
|
var filter service.BackupRecordListInput
|
|
if taskIDValue := strings.TrimSpace(c.Query("taskId")); taskIDValue != "" {
|
|
parsed, ok := parseUintString(taskIDValue)
|
|
if !ok {
|
|
return filter, apperror.BadRequest("BACKUP_RECORD_FILTER_INVALID", "taskId 不合法", nil)
|
|
}
|
|
filter.TaskID = &parsed
|
|
}
|
|
filter.Status = strings.TrimSpace(c.Query("status"))
|
|
if dateFrom := strings.TrimSpace(c.Query("dateFrom")); dateFrom != "" {
|
|
parsed, err := time.Parse(time.RFC3339, dateFrom)
|
|
if err != nil {
|
|
return filter, apperror.BadRequest("BACKUP_RECORD_FILTER_INVALID", "dateFrom 必须为 RFC3339 时间格式", err)
|
|
}
|
|
filter.DateFrom = &parsed
|
|
}
|
|
if dateTo := strings.TrimSpace(c.Query("dateTo")); dateTo != "" {
|
|
parsed, err := time.Parse(time.RFC3339, dateTo)
|
|
if err != nil {
|
|
return filter, apperror.BadRequest("BACKUP_RECORD_FILTER_INVALID", "dateTo 必须为 RFC3339 时间格式", err)
|
|
}
|
|
filter.DateTo = &parsed
|
|
}
|
|
return filter, nil
|
|
}
|
|
|
|
func writeSSEEvent(writer io.Writer, event backup.LogEvent) error {
|
|
payload, err := json.Marshal(event)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = fmt.Fprintf(writer, "event: log\ndata: %s\n\n", payload)
|
|
return err
|
|
}
|
|
|
|
func parseUintString(value string) (uint, bool) {
|
|
parsed, err := strconv.ParseUint(strings.TrimSpace(value), 10, 64)
|
|
if err != nil {
|
|
return 0, false
|
|
}
|
|
return uint(parsed), true
|
|
}
|