mirror of
https://github.com/acepanel/panel.git
synced 2026-02-04 04:22:33 +08:00
231 lines
5.7 KiB
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"))
|
|
}
|