mirror of
https://github.com/1Panel-dev/1Panel.git
synced 2025-09-12 09:34:58 +08:00
feat: 备份目录、网站、数据库功能实现
This commit is contained in:
parent
082ef77a61
commit
fdcfdd7963
8 changed files with 154 additions and 14 deletions
|
@ -15,7 +15,7 @@ type IBackupRepo interface {
|
||||||
Delete(opts ...DBOption) error
|
Delete(opts ...DBOption) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIBackupService() IBackupRepo {
|
func NewIBackupRepo() IBackupRepo {
|
||||||
return &BackupRepo{}
|
return &BackupRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@ type ICommandRepo interface {
|
||||||
Delete(opts ...DBOption) error
|
Delete(opts ...DBOption) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewICommandService() ICommandRepo {
|
func NewICommandRepo() ICommandRepo {
|
||||||
return &CommandRepo{}
|
return &CommandRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ type ICronjobRepo interface {
|
||||||
EndRecords(record model.JobRecords, status, message, records string)
|
EndRecords(record model.JobRecords, status, message, records string)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewICronjobService() ICronjobRepo {
|
func NewICronjobRepo() ICronjobRepo {
|
||||||
return &CronjobRepo{}
|
return &CronjobRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@ type IGroupRepo interface {
|
||||||
Delete(opts ...DBOption) error
|
Delete(opts ...DBOption) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIGroupService() IGroupRepo {
|
func NewIGroupRepo() IGroupRepo {
|
||||||
return &GroupRepo{}
|
return &GroupRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -18,7 +18,7 @@ type IHostRepo interface {
|
||||||
Delete(opts ...DBOption) error
|
Delete(opts ...DBOption) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIHostService() IHostRepo {
|
func NewIHostRepo() IHostRepo {
|
||||||
return &HostRepo{}
|
return &HostRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,7 @@ type IOperationRepo interface {
|
||||||
Delete(opts ...DBOption) error
|
Delete(opts ...DBOption) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIOperationService() IOperationRepo {
|
func NewIOperationRepo() IOperationRepo {
|
||||||
return &OperationRepo{}
|
return &OperationRepo{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,21 +3,29 @@ package service
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/1Panel-dev/1Panel/app/dto"
|
"github.com/1Panel-dev/1Panel/app/dto"
|
||||||
"github.com/1Panel-dev/1Panel/app/model"
|
"github.com/1Panel-dev/1Panel/app/model"
|
||||||
"github.com/1Panel-dev/1Panel/constant"
|
"github.com/1Panel-dev/1Panel/constant"
|
||||||
"github.com/1Panel-dev/1Panel/global"
|
"github.com/1Panel-dev/1Panel/global"
|
||||||
|
"github.com/1Panel-dev/1Panel/utils/cloud_storage"
|
||||||
"github.com/jinzhu/copier"
|
"github.com/jinzhu/copier"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
errRecord = "errRecord"
|
||||||
|
errHandle = "errHandle"
|
||||||
|
)
|
||||||
|
|
||||||
type CronjobService struct{}
|
type CronjobService struct{}
|
||||||
|
|
||||||
type ICronjobService interface {
|
type ICronjobService interface {
|
||||||
|
@ -94,6 +102,8 @@ func (u *CronjobService) Create(cronjobDto dto.CronjobCreate) error {
|
||||||
entryID, err = u.AddShellJob(&cronjob)
|
entryID, err = u.AddShellJob(&cronjob)
|
||||||
case "curl":
|
case "curl":
|
||||||
entryID, err = u.AddCurlJob(&cronjob)
|
entryID, err = u.AddCurlJob(&cronjob)
|
||||||
|
case "directory":
|
||||||
|
entryID, err = u.AddDirectoryJob(&cronjob)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -129,12 +139,13 @@ func (u *CronjobService) AddShellJob(cronjob *model.Cronjob) (int, error) {
|
||||||
cmd := exec.Command(cronjob.Script)
|
cmd := exec.Command(cronjob.Script)
|
||||||
stdout, err := cmd.Output()
|
stdout, err := cmd.Output()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), "ERR_GENERAGE_STDOUT")
|
record.Records = errHandle
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), errHandle)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
record.Records, err = mkdirAndWriteFile(cronjob.ID, cronjob.Name, record.StartTime, stdout)
|
record.Records, err = mkdirAndWriteFile(cronjob, record.StartTime, stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
record.Records = "ERR_CREATE_FILE"
|
record.Records = errRecord
|
||||||
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
||||||
}
|
}
|
||||||
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
||||||
|
@ -159,14 +170,15 @@ func (u *CronjobService) AddCurlJob(cronjob *model.Cronjob) (int, error) {
|
||||||
request, _ := http.NewRequest("GET", cronjob.URL, nil)
|
request, _ := http.NewRequest("GET", cronjob.URL, nil)
|
||||||
response, err := client.Do(request)
|
response, err := client.Do(request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), "")
|
record.Records = errHandle
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), errHandle)
|
||||||
}
|
}
|
||||||
defer response.Body.Close()
|
defer response.Body.Close()
|
||||||
stdout, _ := ioutil.ReadAll(response.Body)
|
stdout, _ := ioutil.ReadAll(response.Body)
|
||||||
|
|
||||||
record.Records, err = mkdirAndWriteFile(cronjob.ID, cronjob.Name, record.StartTime, stdout)
|
record.Records, err = mkdirAndWriteFile(cronjob, record.StartTime, stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
record.Records = "ERR_CREATE_FILE"
|
record.Records = errRecord
|
||||||
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
||||||
}
|
}
|
||||||
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
||||||
|
@ -178,8 +190,60 @@ func (u *CronjobService) AddCurlJob(cronjob *model.Cronjob) (int, error) {
|
||||||
return int(entryID), nil
|
return int(entryID), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func mkdirAndWriteFile(id uint, name string, startTime time.Time, msg []byte) (string, error) {
|
func (u *CronjobService) AddDirectoryJob(cronjob *model.Cronjob) (int, error) {
|
||||||
dir := fmt.Sprintf("/opt/1Panel/task/%s%v/log", name, id)
|
addFunc := func() {
|
||||||
|
record := cronjobRepo.StartRecords(cronjob.ID, "")
|
||||||
|
if len(cronjob.SourceDir) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
message, err := tarWithExclude(cronjob, record.StartTime)
|
||||||
|
if err != nil {
|
||||||
|
record.Records = errHandle
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), errHandle)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
record.Records, err = mkdirAndWriteFile(cronjob, record.StartTime, message)
|
||||||
|
if err != nil {
|
||||||
|
record.Records = errRecord
|
||||||
|
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
||||||
|
}
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
||||||
|
}
|
||||||
|
entryID, err := global.Cron.AddFunc(cronjob.Spec, addFunc)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return int(entryID), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *CronjobService) AddWebSiteJob(cronjob *model.Cronjob) (int, error) {
|
||||||
|
addFunc := func() {
|
||||||
|
record := cronjobRepo.StartRecords(cronjob.ID, "")
|
||||||
|
if len(cronjob.URL) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
message, err := tarWithExclude(cronjob, record.StartTime)
|
||||||
|
if err != nil {
|
||||||
|
record.Records = errHandle
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), errHandle)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
record.Records, err = mkdirAndWriteFile(cronjob, record.StartTime, message)
|
||||||
|
if err != nil {
|
||||||
|
record.Records = errRecord
|
||||||
|
global.LOG.Errorf("save file %s failed, err: %v", record.Records, err)
|
||||||
|
}
|
||||||
|
cronjobRepo.EndRecords(record, constant.StatusSuccess, "", record.Records)
|
||||||
|
}
|
||||||
|
entryID, err := global.Cron.AddFunc(cronjob.Spec, addFunc)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return int(entryID), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func mkdirAndWriteFile(cronjob *model.Cronjob, startTime time.Time, msg []byte) (string, error) {
|
||||||
|
dir := fmt.Sprintf("%s%s/%s-%v", constant.TaskDir, cronjob.Type, cronjob.Name, cronjob.ID)
|
||||||
if _, err := os.Stat(dir); err != nil && os.IsNotExist(err) {
|
if _, err := os.Stat(dir); err != nil && os.IsNotExist(err) {
|
||||||
if err = os.MkdirAll(dir, os.ModePerm); err != nil {
|
if err = os.MkdirAll(dir, os.ModePerm); err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
|
@ -198,6 +262,76 @@ func mkdirAndWriteFile(id uint, name string, startTime time.Time, msg []byte) (s
|
||||||
return path, nil
|
return path, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func tarWithExclude(cronjob *model.Cronjob, startTime time.Time) ([]byte, error) {
|
||||||
|
varMaps, targetdir, err := loadTargetInfo(cronjob)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("load target dir failed, err: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
excludes := strings.Split(cronjob.ExclusionRules, ";")
|
||||||
|
name := fmt.Sprintf("%s/%s.tar.gz", targetdir, startTime.Format("20060102150405"))
|
||||||
|
exStr := []string{"-zcvPf", name}
|
||||||
|
for _, exclude := range excludes {
|
||||||
|
exStr = append(exStr, "--exclude")
|
||||||
|
exStr = append(exStr, exclude)
|
||||||
|
}
|
||||||
|
exStr = append(exStr, cronjob.SourceDir)
|
||||||
|
cmd := exec.Command("tar", exStr...)
|
||||||
|
stdout, err := cmd.Output()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("tar zcvPf failed, err: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if varMaps["type"] != "LOCAL" {
|
||||||
|
backClient, err := cloud_storage.NewCloudStorageClient(varMaps)
|
||||||
|
if err != nil {
|
||||||
|
return stdout, fmt.Errorf("new cloud storage client failed, err: %v", err)
|
||||||
|
}
|
||||||
|
isOK, err := backClient.Upload(name, strings.Replace(name, constant.TmpDir, "", -1))
|
||||||
|
if !isOK {
|
||||||
|
return nil, fmt.Errorf("cloud storage upload failed, err: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return stdout, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func loadTargetInfo(cronjob *model.Cronjob) (map[string]interface{}, string, error) {
|
||||||
|
backup, err := backupRepo.Get(commonRepo.WithByID(uint(cronjob.TargetDirID)))
|
||||||
|
if err != nil {
|
||||||
|
return nil, "", err
|
||||||
|
}
|
||||||
|
varMap := make(map[string]interface{})
|
||||||
|
if err := json.Unmarshal([]byte(backup.Vars), &varMap); err != nil {
|
||||||
|
return nil, "", err
|
||||||
|
}
|
||||||
|
dir := ""
|
||||||
|
if backup.Type != "LOCAL" {
|
||||||
|
varMap["type"] = backup.Type
|
||||||
|
varMap["bucket"] = backup.Bucket
|
||||||
|
switch backup.Type {
|
||||||
|
case constant.Sftp:
|
||||||
|
varMap["password"] = backup.Credential
|
||||||
|
case constant.OSS, constant.S3, constant.MinIo:
|
||||||
|
varMap["secretKey"] = backup.Credential
|
||||||
|
}
|
||||||
|
dir = fmt.Sprintf("%s%s/%s", constant.TmpDir, cronjob.Type, cronjob.Name)
|
||||||
|
} else {
|
||||||
|
if _, ok := varMap["dir"]; !ok {
|
||||||
|
return nil, "", errors.New("load local backup dir failed")
|
||||||
|
}
|
||||||
|
dir = fmt.Sprintf("%v/%s/%s", varMap["dir"], cronjob.Type, cronjob.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := os.Stat(dir); err != nil && os.IsNotExist(err) {
|
||||||
|
if err = os.MkdirAll(dir, os.ModePerm); err != nil {
|
||||||
|
if err != nil {
|
||||||
|
return nil, "", fmt.Errorf("mkdir %s failed, err: %v", dir, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return varMap, dir, nil
|
||||||
|
}
|
||||||
|
|
||||||
func loadSpec(cronjob model.Cronjob) string {
|
func loadSpec(cronjob model.Cronjob) string {
|
||||||
switch cronjob.SpecType {
|
switch cronjob.SpecType {
|
||||||
case "perMonth":
|
case "perMonth":
|
||||||
|
|
6
backend/constant/cronjob.go
Normal file
6
backend/constant/cronjob.go
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
package constant
|
||||||
|
|
||||||
|
const (
|
||||||
|
TmpDir = "/opt/1Panel/task/tmp/"
|
||||||
|
TaskDir = "/opt/1Panel/task/"
|
||||||
|
)
|
Loading…
Add table
Reference in a new issue