2
0
mirror of https://github.com/acepanel/panel.git synced 2026-02-04 04:22:33 +08:00
Files
panel/internal/data/cron.go

231 lines
5.7 KiB
Go

package data
import (
"context"
"errors"
"fmt"
"log/slog"
"path/filepath"
"github.com/leonelquinteros/gotext"
"github.com/libtnb/utils/str"
"gorm.io/gorm"
"github.com/acepanel/panel/internal/app"
"github.com/acepanel/panel/internal/biz"
"github.com/acepanel/panel/internal/http/request"
"github.com/acepanel/panel/pkg/io"
"github.com/acepanel/panel/pkg/os"
"github.com/acepanel/panel/pkg/shell"
"github.com/acepanel/panel/pkg/systemctl"
)
type cronRepo struct {
t *gotext.Locale
db *gorm.DB
log *slog.Logger
}
func NewCronRepo(t *gotext.Locale, db *gorm.DB, log *slog.Logger) biz.CronRepo {
return &cronRepo{
t: t,
db: db,
log: log,
}
}
func (r *cronRepo) Count() (int64, error) {
var count int64
if err := r.db.Model(&biz.Cron{}).Count(&count).Error; err != nil {
return 0, err
}
return count, nil
}
func (r *cronRepo) List(page, limit uint) ([]*biz.Cron, int64, error) {
cron := make([]*biz.Cron, 0)
var total int64
err := r.db.Model(&biz.Cron{}).Order("id desc").Count(&total).Offset(int((page - 1) * limit)).Limit(int(limit)).Find(&cron).Error
return cron, total, err
}
func (r *cronRepo) Get(id uint) (*biz.Cron, error) {
cron := new(biz.Cron)
if err := r.db.Where("id = ?", id).First(cron).Error; err != nil {
return nil, err
}
return cron, nil
}
func (r *cronRepo) Create(ctx context.Context, req *request.CronCreate) error {
var script string
if req.Type == "backup" {
if req.BackupType == "website" {
script = fmt.Sprintf(`#!/bin/bash
export PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:$PATH
acepanel backup website -n '%s' -s '%d'
acepanel backup clear -t website -f '%s' -k '%d' -s '%d'
`, req.Target, req.BackupStorage, req.Target, req.Keep, req.BackupStorage)
}
if req.BackupType == "mysql" || req.BackupType == "postgres" {
script = fmt.Sprintf(`#!/bin/bash
export PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:$PATH
acepanel backup database -t '%s' -n '%s' -s '%d'
acepanel backup clear -t '%s' -f '%s' -k '%d' -s '%d'
`, req.BackupType, req.Target, req.BackupStorage, req.BackupType, req.Target, req.Keep, req.BackupStorage)
}
}
if req.Type == "cutoff" {
script = fmt.Sprintf(`#!/bin/bash
export PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:$PATH
acepanel cutoff website -n '%s'
acepanel cutoff clear -t website -n '%s' -k '%d'
`, req.Target, req.Target, req.Keep)
}
if req.Type == "shell" {
script = req.Script
}
shellDir := fmt.Sprintf("%s/server/cron/", app.Root)
shellLogDir := fmt.Sprintf("%s/server/cron/logs/", app.Root)
shellFile := str.Random(16)
if err := io.Write(filepath.Join(shellDir, shellFile+".sh"), script, 0700); err != nil {
return errors.New(err.Error())
}
// 编码转换
_, _ = shell.Execf("dos2unix %s%s.sh", shellDir, shellFile)
cron := new(biz.Cron)
cron.Name = req.Name
cron.Type = req.Type
cron.Status = true
cron.Time = req.Time
cron.Shell = shellDir + shellFile + ".sh"
cron.Log = shellLogDir + shellFile + ".log"
if err := r.db.Create(cron).Error; err != nil {
return err
}
if err := r.addToSystem(cron); err != nil {
return err
}
// 记录日志
r.log.Info("cron created", slog.String("type", biz.OperationTypeCron), slog.Uint64("operator_id", getOperatorID(ctx)), slog.String("name", req.Name), slog.String("cron_type", req.Type))
return nil
}
func (r *cronRepo) Update(ctx context.Context, req *request.CronUpdate) error {
cron, err := r.Get(req.ID)
if err != nil {
return err
}
cron.Time = req.Time
cron.Name = req.Name
if err = r.db.Save(cron).Error; err != nil {
return err
}
if err = io.Write(cron.Shell, req.Script, 0700); err != nil {
return err
}
if out, err := shell.Execf("dos2unix %s", cron.Shell); err != nil {
return errors.New(out)
}
if err = r.deleteFromSystem(cron); err != nil {
return err
}
if cron.Status {
if err = r.addToSystem(cron); err != nil {
return err
}
}
// 记录日志
r.log.Info("cron updated", slog.String("type", biz.OperationTypeCron), slog.Uint64("operator_id", getOperatorID(ctx)), slog.Uint64("id", uint64(req.ID)), slog.String("name", cron.Name))
return nil
}
func (r *cronRepo) Delete(ctx context.Context, id uint) error {
cron, err := r.Get(id)
if err != nil {
return err
}
if err = r.deleteFromSystem(cron); err != nil {
return err
}
if err = io.Remove(cron.Shell); err != nil {
return err
}
if err = r.db.Delete(cron).Error; err != nil {
return err
}
// 记录日志
r.log.Info("cron deleted", slog.String("type", biz.OperationTypeCron), slog.Uint64("operator_id", getOperatorID(ctx)), slog.Uint64("id", uint64(id)), slog.String("name", cron.Name))
return nil
}
func (r *cronRepo) Status(id uint, status bool) error {
cron, err := r.Get(id)
if err != nil {
return err
}
if err = r.deleteFromSystem(cron); err != nil {
return err
}
if status {
if err = r.addToSystem(cron); err != nil {
return err
}
}
cron.Status = status
return r.db.Save(cron).Error
}
// addToSystem 添加到系统
func (r *cronRepo) addToSystem(cron *biz.Cron) error {
if _, err := shell.Execf(`( crontab -l; echo "%s %s >> %s 2>&1" ) | sort - | uniq - | crontab -`, cron.Time, cron.Shell, cron.Log); err != nil {
return err
}
return r.restartCron()
}
// deleteFromSystem 从系统中删除
func (r *cronRepo) deleteFromSystem(cron *biz.Cron) error {
if _, err := shell.Execf(`( crontab -l | grep -v -F "%s %s >> %s 2>&1" ) | crontab -`, cron.Time, cron.Shell, cron.Log); err != nil {
return err
}
return r.restartCron()
}
// restartCron 重启 cron 服务
func (r *cronRepo) restartCron() error {
if os.IsRHEL() {
return systemctl.Restart("crond")
}
if os.IsDebian() || os.IsUbuntu() {
return systemctl.Restart("cron")
}
return errors.New(r.t.Get("unsupported system"))
}