From 64ad829e3cd380a3fa0154a74b132b4a6855a40f Mon Sep 17 00:00:00 2001
From: ssongliu <73214554+ssongliu@users.noreply.github.com>
Date: Fri, 1 Dec 2023 17:50:09 +0800
Subject: [PATCH] =?UTF-8?q?feat:=20=E5=A4=87=E4=BB=BD=E5=88=97=E8=A1=A8?=
=?UTF-8?q?=E6=98=BE=E7=A4=BA=E6=96=87=E4=BB=B6=E5=A4=87=E4=BB=BD=E6=96=87?=
=?UTF-8?q?=E4=BB=B6=E5=A4=A7=E5=B0=8F=20(#3134)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
Refs #2968 #1922
---
backend/app/dto/backup.go | 1 +
backend/app/service/backup.go | 45 ++++-
backend/app/service/cronjob_helper.go | 20 +-
backend/utils/cloud_storage/client/cos.go | 133 +++++--------
backend/utils/cloud_storage/client/helper.go | 20 ++
backend/utils/cloud_storage/client/kodo.go | 91 +++------
backend/utils/cloud_storage/client/minio.go | 146 ++++++---------
.../utils/cloud_storage/client/onedrive.go | 94 ++++++----
backend/utils/cloud_storage/client/oss.go | 113 +++++-------
backend/utils/cloud_storage/client/s3.go | 166 ++++++-----------
backend/utils/cloud_storage/client/sftp.go | 174 ++++++++++--------
backend/utils/cloud_storage/client/webdav.go | 48 ++---
.../cloud_storage/cloud_storage_client.go | 2 +
frontend/src/components/backup/index.vue | 23 ++-
.../setting/backup-account/operate/index.vue | 2 +-
15 files changed, 491 insertions(+), 587 deletions(-)
create mode 100644 backend/utils/cloud_storage/client/helper.go
diff --git a/backend/app/dto/backup.go b/backend/app/dto/backup.go
index 6dd3ce0ec..60486c9f7 100644
--- a/backend/app/dto/backup.go
+++ b/backend/app/dto/backup.go
@@ -52,6 +52,7 @@ type BackupRecords struct {
BackupType string `json:"backupType"`
FileDir string `json:"fileDir"`
FileName string `json:"fileName"`
+ Size int64 `json:"size"`
}
type DownloadRecord struct {
diff --git a/backend/app/service/backup.go b/backend/app/service/backup.go
index 1986ed57a..a884bd657 100644
--- a/backend/app/service/backup.go
+++ b/backend/app/service/backup.go
@@ -81,15 +81,46 @@ func (u *BackupService) SearchRecordsWithPage(search dto.RecordSearch) (int64, [
commonRepo.WithByType(search.Type),
backupRepo.WithByDetailName(search.DetailName),
)
- var dtobas []dto.BackupRecords
- for _, group := range records {
+
+ var datas []dto.BackupRecords
+ clientMap := make(map[string]loadSizeHelper)
+ for i := 0; i < len(records); i++ {
var item dto.BackupRecords
- if err := copier.Copy(&item, &group); err != nil {
+ if err := copier.Copy(&item, &records[i]); err != nil {
return 0, nil, errors.WithMessage(constant.ErrStructTransform, err.Error())
}
- dtobas = append(dtobas, item)
+ itemPath := path.Join(records[i].FileDir, records[i].FileName)
+ if records[i].Source == "LOCAL" {
+ fileInfo, _ := os.Stat(itemPath)
+ item.Size = fileInfo.Size()
+ datas = append(datas, item)
+ continue
+ }
+ if _, ok := clientMap[records[i].Source]; !ok {
+ backup, err := backupRepo.Get(commonRepo.WithByType(records[i].Source))
+ if err != nil {
+ global.LOG.Errorf("load backup model %s from db failed, err: %v", records[i].Source, err)
+ return total, datas, err
+ }
+ client, err := u.NewClient(&backup)
+ if err != nil {
+ global.LOG.Errorf("load backup client %s from db failed, err: %v", records[i].Source, err)
+ return total, datas, err
+ }
+ item.Size, _ = client.Size(path.Join(strings.TrimLeft(backup.BackupPath, "/"), itemPath))
+ datas = append(datas, item)
+ clientMap[records[i].Source] = loadSizeHelper{backupPath: strings.TrimLeft(backup.BackupPath, "/"), client: client}
+ continue
+ }
+ item.Size, _ = clientMap[records[i].Source].client.Size(path.Join(clientMap[records[i].Source].backupPath, itemPath))
+ datas = append(datas, item)
}
- return total, dtobas, err
+ return total, datas, err
+}
+
+type loadSizeHelper struct {
+ backupPath string
+ client cloud_storage.CloudStorageClient
}
func (u *BackupService) LoadOneDriveInfo() (string, error) {
@@ -139,9 +170,7 @@ func (u *BackupService) DownloadRecord(info dto.DownloadRecord) (string, error)
}
srcPath := fmt.Sprintf("%s/%s", info.FileDir, info.FileName)
if len(backup.BackupPath) != 0 {
- itemPath := strings.TrimPrefix(backup.BackupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- srcPath = itemPath + srcPath
+ srcPath = path.Join(strings.TrimPrefix(backup.BackupPath, "/"), srcPath)
}
if exist, _ := backClient.Exist(srcPath); exist {
isOK, err := backClient.Download(srcPath, targetPath)
diff --git a/backend/app/service/cronjob_helper.go b/backend/app/service/cronjob_helper.go
index 903d3b621..c29f0e9f2 100644
--- a/backend/app/service/cronjob_helper.go
+++ b/backend/app/service/cronjob_helper.go
@@ -157,9 +157,7 @@ func (u *CronjobService) handleBackup(cronjob *model.Cronjob, startTime time.Tim
return "", err
}
if len(backup.BackupPath) != 0 {
- itemPath := strings.TrimPrefix(backup.BackupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- itemFileDir = itemPath + itemFileDir
+ itemFileDir = path.Join(strings.TrimPrefix(backup.BackupPath, "/"), itemFileDir)
}
if _, err = client.Upload(backupDir+"/"+fileName, itemFileDir+"/"+fileName); err != nil {
return "", err
@@ -193,9 +191,7 @@ func (u *CronjobService) HandleRmExpired(backType, backupPath, localDir string,
fileItem := file
if cronjob.KeepLocal {
if len(backupPath) != 0 {
- itemPath := strings.TrimPrefix(backupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- fileItem = itemPath + strings.TrimPrefix(file, localDir+"/")
+ fileItem = path.Join(strings.TrimPrefix(backupPath, "/") + strings.TrimPrefix(file, localDir+"/"))
} else {
fileItem = strings.TrimPrefix(file, localDir+"/")
}
@@ -332,9 +328,7 @@ func (u *CronjobService) handleDatabase(cronjob model.Cronjob, backup model.Back
}()
}
if len(backup.BackupPath) != 0 {
- itemPath := strings.TrimPrefix(backup.BackupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- itemFileDir = itemPath + itemFileDir
+ itemFileDir = path.Join(strings.TrimPrefix(backup.BackupPath, "/"), itemFileDir)
}
if _, err = client.Upload(backupDir+"/"+record.FileName, itemFileDir+"/"+record.FileName); err != nil {
return paths, err
@@ -475,9 +469,7 @@ func (u *CronjobService) handleApp(cronjob model.Cronjob, backup model.BackupAcc
}()
}
if len(backup.BackupPath) != 0 {
- itemPath := strings.TrimPrefix(backup.BackupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- itemFileDir = itemPath + itemFileDir
+ itemFileDir = path.Join(strings.TrimPrefix(backup.BackupPath, "/"), itemFileDir)
}
if _, err = client.Upload(backupDir+"/"+record.FileName, itemFileDir+"/"+record.FileName); err != nil {
return paths, err
@@ -551,9 +543,7 @@ func (u *CronjobService) handleWebsite(cronjob model.Cronjob, backup model.Backu
}()
}
if len(backup.BackupPath) != 0 {
- itemPath := strings.TrimPrefix(backup.BackupPath, "/")
- itemPath = strings.TrimSuffix(itemPath, "/") + "/"
- itemFileDir = itemPath + itemFileDir
+ itemFileDir = path.Join(strings.TrimPrefix(backup.BackupPath, "/"), itemFileDir)
}
if _, err = client.Upload(backupDir+"/"+record.FileName, itemFileDir+"/"+record.FileName); err != nil {
return paths, err
diff --git a/backend/utils/cloud_storage/client/cos.go b/backend/utils/cloud_storage/client/cos.go
index 8eef7cbe1..450cb1a16 100644
--- a/backend/utils/cloud_storage/client/cos.go
+++ b/backend/utils/cloud_storage/client/cos.go
@@ -7,44 +7,24 @@ import (
"net/url"
"os"
- "github.com/1Panel-dev/1Panel/backend/constant"
cosSDK "github.com/tencentyun/cos-go-sdk-v5"
)
type cosClient struct {
- region string
- accessKey string
- secretKey string
- scType string
- Vars map[string]interface{}
- client *cosSDK.Client
+ scType string
+ client *cosSDK.Client
+ clientWithBucket *cosSDK.Client
}
func NewCosClient(vars map[string]interface{}) (*cosClient, error) {
- var accessKey string
- var secretKey string
- var scType string
- var region string
- if _, ok := vars["region"]; ok {
- region = vars["region"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["accessKey"]; ok {
- accessKey = vars["accessKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["scType"]; ok {
- scType = vars["scType"].(string)
- } else {
+ region := loadParamFromVars("region", true, vars)
+ accessKey := loadParamFromVars("accessKey", true, vars)
+ secretKey := loadParamFromVars("secretKey", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
+ scType := loadParamFromVars("scType", true, vars)
+ if len(scType) == 0 {
scType = "Standard"
}
- if _, ok := vars["secretKey"]; ok {
- secretKey = vars["secretKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
u, _ := url.Parse(fmt.Sprintf("https://cos.%s.myqcloud.com", region))
b := &cosSDK.BaseURL{BucketURL: u}
@@ -55,11 +35,23 @@ func NewCosClient(vars map[string]interface{}) (*cosClient, error) {
},
})
- return &cosClient{Vars: vars, client: client, accessKey: accessKey, secretKey: secretKey, scType: scType, region: region}, nil
+ if len(bucket) != 0 {
+ u2, _ := url.Parse(fmt.Sprintf("https://%s.cos.%s.myqcloud.com", bucket, region))
+ b2 := &cosSDK.BaseURL{BucketURL: u2}
+ clientWithBucket := cosSDK.NewClient(b2, &http.Client{
+ Transport: &cosSDK.AuthorizationTransport{
+ SecretID: accessKey,
+ SecretKey: secretKey,
+ },
+ })
+ return &cosClient{client: client, clientWithBucket: clientWithBucket, scType: scType}, nil
+ }
+
+ return &cosClient{client: client, clientWithBucket: nil, scType: scType}, nil
}
-func (cos cosClient) ListBuckets() ([]interface{}, error) {
- buckets, _, err := cos.client.Service.Get(context.Background())
+func (c cosClient) ListBuckets() ([]interface{}, error) {
+ buckets, _, err := c.client.Service.Get(context.Background())
if err != nil {
return nil, err
}
@@ -70,34 +62,33 @@ func (cos cosClient) ListBuckets() ([]interface{}, error) {
return datas, nil
}
-func (cos cosClient) Exist(path string) (bool, error) {
- client, err := cos.newClientWithBucket()
- if err != nil {
- return false, err
- }
- exist, err := client.Object.IsExist(context.Background(), path)
+func (c cosClient) Exist(path string) (bool, error) {
+ exist, err := c.clientWithBucket.Object.IsExist(context.Background(), path)
if err != nil {
return false, err
}
return exist, nil
}
-func (cos cosClient) Delete(path string) (bool, error) {
- client, err := cos.newClientWithBucket()
+func (c cosClient) Size(path string) (int64, error) {
+ data, _, err := c.clientWithBucket.Bucket.Get(context.Background(), &cosSDK.BucketGetOptions{Prefix: path})
if err != nil {
- return false, err
+ return 0, err
}
- if _, err := client.Object.Delete(context.Background(), path); err != nil {
+ if len(data.Contents) == 0 {
+ return 0, fmt.Errorf("no such file %s", path)
+ }
+ return data.Contents[0].Size, nil
+}
+
+func (c cosClient) Delete(path string) (bool, error) {
+ if _, err := c.clientWithBucket.Object.Delete(context.Background(), path); err != nil {
return false, err
}
return true, nil
}
-func (cos cosClient) Upload(src, target string) (bool, error) {
- client, err := cos.newClientWithBucket()
- if err != nil {
- return false, err
- }
+func (c cosClient) Upload(src, target string) (bool, error) {
fileInfo, err := os.Stat(src)
if err != nil {
return false, err
@@ -107,22 +98,22 @@ func (cos cosClient) Upload(src, target string) (bool, error) {
OptIni: &cosSDK.InitiateMultipartUploadOptions{
ACLHeaderOptions: nil,
ObjectPutHeaderOptions: &cosSDK.ObjectPutHeaderOptions{
- XCosStorageClass: cos.scType,
+ XCosStorageClass: c.scType,
},
},
PartSize: 200,
}
- if _, _, err := client.Object.MultiUpload(
+ if _, _, err := c.clientWithBucket.Object.MultiUpload(
context.Background(), target, src, opt,
); err != nil {
return false, err
}
return true, nil
}
- if _, err := client.Object.PutFromFile(context.Background(), target, src, &cosSDK.ObjectPutOptions{
+ if _, err := c.clientWithBucket.Object.PutFromFile(context.Background(), target, src, &cosSDK.ObjectPutOptions{
ACLHeaderOptions: nil,
ObjectPutHeaderOptions: &cosSDK.ObjectPutHeaderOptions{
- XCosStorageClass: cos.scType,
+ XCosStorageClass: c.scType,
},
}); err != nil {
return false, err
@@ -130,31 +121,15 @@ func (cos cosClient) Upload(src, target string) (bool, error) {
return true, nil
}
-func (cos cosClient) Download(src, target string) (bool, error) {
- client, err := cos.newClientWithBucket()
- if err != nil {
- return false, err
- }
- if _, err := client.Object.Download(context.Background(), src, target, &cosSDK.MultiDownloadOptions{}); err != nil {
+func (c cosClient) Download(src, target string) (bool, error) {
+ if _, err := c.clientWithBucket.Object.Download(context.Background(), src, target, &cosSDK.MultiDownloadOptions{}); err != nil {
return false, err
}
return true, nil
}
-func (cos *cosClient) GetBucket() (string, error) {
- if _, ok := cos.Vars["bucket"]; ok {
- return cos.Vars["bucket"].(string), nil
- } else {
- return "", constant.ErrInvalidParams
- }
-}
-
-func (cos cosClient) ListObjects(prefix string) ([]string, error) {
- client, err := cos.newClientWithBucket()
- if err != nil {
- return nil, err
- }
- datas, _, err := client.Bucket.Get(context.Background(), &cosSDK.BucketGetOptions{Prefix: prefix})
+func (c cosClient) ListObjects(prefix string) ([]string, error) {
+ datas, _, err := c.clientWithBucket.Bucket.Get(context.Background(), &cosSDK.BucketGetOptions{Prefix: prefix})
if err != nil {
return nil, err
}
@@ -165,19 +140,3 @@ func (cos cosClient) ListObjects(prefix string) ([]string, error) {
}
return result, nil
}
-
-func (cos cosClient) newClientWithBucket() (*cosSDK.Client, error) {
- bucket, err := cos.GetBucket()
- if err != nil {
- return nil, err
- }
- u, _ := url.Parse(fmt.Sprintf("https://%s.cos.%s.myqcloud.com", bucket, cos.region))
- b := &cosSDK.BaseURL{BucketURL: u}
- client := cosSDK.NewClient(b, &http.Client{
- Transport: &cosSDK.AuthorizationTransport{
- SecretID: cos.accessKey,
- SecretKey: cos.secretKey,
- },
- })
- return client, nil
-}
diff --git a/backend/utils/cloud_storage/client/helper.go b/backend/utils/cloud_storage/client/helper.go
new file mode 100644
index 000000000..0c05dc5ac
--- /dev/null
+++ b/backend/utils/cloud_storage/client/helper.go
@@ -0,0 +1,20 @@
+package client
+
+import (
+ "fmt"
+
+ "github.com/1Panel-dev/1Panel/backend/global"
+)
+
+func loadParamFromVars(key string, isString bool, vars map[string]interface{}) string {
+ if _, ok := vars[key]; !ok {
+ if key != "bucket" {
+ global.LOG.Errorf("load param %s from vars failed, err: not exist!", key)
+ }
+ return ""
+ }
+ if isString {
+ return vars[key].(string)
+ }
+ return fmt.Sprintf("%v", vars[key].(float64))
+}
diff --git a/backend/utils/cloud_storage/client/kodo.go b/backend/utils/cloud_storage/client/kodo.go
index 872d90af1..80a4fb64e 100644
--- a/backend/utils/cloud_storage/client/kodo.go
+++ b/backend/utils/cloud_storage/client/kodo.go
@@ -4,45 +4,35 @@ import (
"context"
"time"
- "github.com/1Panel-dev/1Panel/backend/constant"
"github.com/1Panel-dev/1Panel/backend/utils/files"
"github.com/qiniu/go-sdk/v7/auth"
- "github.com/qiniu/go-sdk/v7/auth/qbox"
"github.com/qiniu/go-sdk/v7/storage"
)
type kodoClient struct {
- accessKey string
- secretKey string
- Vars map[string]interface{}
- client *storage.BucketManager
+ bucket string
+ domain string
+ auth *auth.Credentials
+ client *storage.BucketManager
}
func NewKodoClient(vars map[string]interface{}) (*kodoClient, error) {
- var accessKey string
- var secretKey string
- if _, ok := vars["accessKey"]; ok {
- accessKey = vars["accessKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["secretKey"]; ok {
- secretKey = vars["secretKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
+ accessKey := loadParamFromVars("accessKey", true, vars)
+ secretKey := loadParamFromVars("secretKey", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
+ domain := loadParamFromVars("domain", true, vars)
- conn := qbox.NewMac(accessKey, secretKey)
+ conn := auth.New(accessKey, secretKey)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(conn, &cfg)
- return &kodoClient{Vars: vars, client: bucketManager, accessKey: accessKey, secretKey: secretKey}, nil
+ return &kodoClient{client: bucketManager, auth: conn, bucket: bucket, domain: domain}, nil
}
-func (kodo kodoClient) ListBuckets() ([]interface{}, error) {
- buckets, err := kodo.client.Buckets(true)
+func (k kodoClient) ListBuckets() ([]interface{}, error) {
+ buckets, err := k.client.Buckets(true)
if err != nil {
return nil, err
}
@@ -53,38 +43,33 @@ func (kodo kodoClient) ListBuckets() ([]interface{}, error) {
return datas, nil
}
-func (kodo kodoClient) Exist(path string) (bool, error) {
- bucket, err := kodo.GetBucket()
- if err != nil {
- return false, err
- }
- if _, err := kodo.client.Stat(bucket, path); err != nil {
+func (k kodoClient) Exist(path string) (bool, error) {
+ if _, err := k.client.Stat(k.bucket, path); err != nil {
return false, err
}
return true, nil
}
-func (kodo kodoClient) Delete(path string) (bool, error) {
- bucket, err := kodo.GetBucket()
+func (k kodoClient) Size(path string) (int64, error) {
+ file, err := k.client.Stat(k.bucket, path)
if err != nil {
- return false, err
+ return 0, err
}
- if err := kodo.client.Delete(bucket, path); err != nil {
+ return file.Fsize, nil
+}
+
+func (k kodoClient) Delete(path string) (bool, error) {
+ if err := k.client.Delete(k.bucket, path); err != nil {
return false, err
}
return true, nil
}
-func (kodo kodoClient) Upload(src, target string) (bool, error) {
- bucket, err := kodo.GetBucket()
- if err != nil {
- return false, err
- }
+func (k kodoClient) Upload(src, target string) (bool, error) {
putPolicy := storage.PutPolicy{
- Scope: bucket,
+ Scope: k.bucket,
}
- mac := qbox.NewMac(kodo.accessKey, kodo.secretKey)
- upToken := putPolicy.UploadToken(mac)
+ upToken := putPolicy.UploadToken(k.auth)
cfg := storage.Config{UseHTTPS: true, UseCdnDomains: false}
resumeUploader := storage.NewResumeUploaderV2(&cfg)
ret := storage.PutRet{}
@@ -95,14 +80,9 @@ func (kodo kodoClient) Upload(src, target string) (bool, error) {
return true, nil
}
-func (kodo kodoClient) Download(src, target string) (bool, error) {
- mac := auth.New(kodo.accessKey, kodo.secretKey)
- if _, ok := kodo.Vars["domain"]; !ok {
- return false, constant.ErrInvalidParams
- }
- domain := kodo.Vars["domain"].(string)
+func (k kodoClient) Download(src, target string) (bool, error) {
deadline := time.Now().Add(time.Second * 3600).Unix()
- privateAccessURL := storage.MakePrivateURL(mac, domain, src, deadline)
+ privateAccessURL := storage.MakePrivateURL(k.auth, k.domain, src, deadline)
fo := files.NewFileOp()
if err := fo.DownloadFile(privateAccessURL, target); err != nil {
@@ -111,24 +91,11 @@ func (kodo kodoClient) Download(src, target string) (bool, error) {
return true, nil
}
-func (kodo *kodoClient) GetBucket() (string, error) {
- if _, ok := kodo.Vars["bucket"]; ok {
- return kodo.Vars["bucket"].(string), nil
- } else {
- return "", constant.ErrInvalidParams
- }
-}
-
-func (kodo kodoClient) ListObjects(prefix string) ([]string, error) {
- bucket, err := kodo.GetBucket()
- if err != nil {
- return nil, constant.ErrInvalidParams
- }
-
+func (k kodoClient) ListObjects(prefix string) ([]string, error) {
var result []string
marker := ""
for {
- entries, _, nextMarker, hashNext, err := kodo.client.ListFiles(bucket, prefix, "", marker, 1000)
+ entries, _, nextMarker, hashNext, err := k.client.ListFiles(k.bucket, prefix, "", marker, 1000)
if err != nil {
return nil, err
}
diff --git a/backend/utils/cloud_storage/client/minio.go b/backend/utils/cloud_storage/client/minio.go
index 6264507e7..c52ae3ffb 100644
--- a/backend/utils/cloud_storage/client/minio.go
+++ b/backend/utils/cloud_storage/client/minio.go
@@ -14,29 +14,15 @@ import (
)
type minIoClient struct {
- Vars map[string]interface{}
+ bucket string
client *minio.Client
}
func NewMinIoClient(vars map[string]interface{}) (*minIoClient, error) {
- var endpoint string
- var accessKeyID string
- var secretAccessKey string
- if _, ok := vars["endpoint"]; ok {
- endpoint = vars["endpoint"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["accessKey"]; ok {
- accessKeyID = vars["accessKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["secretKey"]; ok {
- secretAccessKey = vars["secretKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
+ endpoint := loadParamFromVars("endpoint", true, vars)
+ accessKeyID := loadParamFromVars("accessKey", true, vars)
+ secretAccessKey := loadParamFromVars("secretKey", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
ssl := strings.Split(endpoint, ":")[0]
if len(ssl) == 0 || (ssl != "https" && ssl != "http") {
return nil, constant.ErrInvalidParams
@@ -59,14 +45,11 @@ func NewMinIoClient(vars map[string]interface{}) (*minIoClient, error) {
if err != nil {
return nil, err
}
- return &minIoClient{
- client: client,
- Vars: vars,
- }, nil
+ return &minIoClient{bucket: bucket, client: client}, nil
}
-func (minIo minIoClient) ListBuckets() ([]interface{}, error) {
- buckets, err := minIo.client.ListBuckets(context.Background())
+func (m minIoClient) ListBuckets() ([]interface{}, error) {
+ buckets, err := m.client.ListBuckets(context.Background())
if err != nil {
return nil, err
}
@@ -77,49 +60,44 @@ func (minIo minIoClient) ListBuckets() ([]interface{}, error) {
return result, err
}
-func (minIo minIoClient) Exist(path string) (bool, error) {
- if _, ok := minIo.Vars["bucket"]; ok {
- _, err := minIo.client.GetObject(context.Background(), minIo.Vars["bucket"].(string), path, minio.GetObjectOptions{})
- if err != nil {
- return false, err
- }
- return true, nil
- } else {
- return false, constant.ErrInvalidParams
+func (m minIoClient) Exist(path string) (bool, error) {
+ if _, err := m.client.GetObject(context.Background(), m.bucket, path, minio.GetObjectOptions{}); err != nil {
+ return false, err
}
+ return true, nil
}
-func (minIo minIoClient) Delete(path string) (bool, error) {
- if _, ok := minIo.Vars["bucket"]; ok {
- object, err := minIo.client.GetObject(context.Background(), minIo.Vars["bucket"].(string), path, minio.GetObjectOptions{})
- if err != nil {
- return false, err
- }
- info, err := object.Stat()
- if err != nil {
- return false, err
- }
- err = minIo.client.RemoveObject(context.Background(), minIo.Vars["bucket"].(string), path, minio.RemoveObjectOptions{
- GovernanceBypass: true,
- VersionID: info.VersionID,
- })
- if err != nil {
- return false, err
- }
- return true, nil
- } else {
- return false, constant.ErrInvalidParams
+func (m minIoClient) Size(path string) (int64, error) {
+ obj, err := m.client.GetObject(context.Background(), m.bucket, path, minio.GetObjectOptions{})
+ if err != nil {
+ return 0, err
}
+ file, err := obj.Stat()
+ if err != nil {
+ return 0, err
+ }
+ return file.Size, nil
}
-func (minIo minIoClient) Upload(src, target string) (bool, error) {
- var bucket string
- if _, ok := minIo.Vars["bucket"]; ok {
- bucket = minIo.Vars["bucket"].(string)
- } else {
- return false, constant.ErrInvalidParams
+func (m minIoClient) Delete(path string) (bool, error) {
+ object, err := m.client.GetObject(context.Background(), m.bucket, path, minio.GetObjectOptions{})
+ if err != nil {
+ return false, err
}
+ info, err := object.Stat()
+ if err != nil {
+ return false, err
+ }
+ if err = m.client.RemoveObject(context.Background(), m.bucket, path, minio.RemoveObjectOptions{
+ GovernanceBypass: true,
+ VersionID: info.VersionID,
+ }); err != nil {
+ return false, err
+ }
+ return true, nil
+}
+func (m minIoClient) Upload(src, target string) (bool, error) {
file, err := os.Open(src)
if err != nil {
return false, err
@@ -130,52 +108,36 @@ func (minIo minIoClient) Upload(src, target string) (bool, error) {
if err != nil {
return false, err
}
- _, err = minIo.client.PutObject(context.Background(), bucket, target, file, fileStat.Size(), minio.PutObjectOptions{ContentType: "application/octet-stream"})
+ _, err = m.client.PutObject(context.Background(), m.bucket, target, file, fileStat.Size(), minio.PutObjectOptions{ContentType: "application/octet-stream"})
if err != nil {
return false, err
}
return true, nil
}
-func (minIo minIoClient) Download(src, target string) (bool, error) {
- if _, ok := minIo.Vars["bucket"]; ok {
- object, err := minIo.client.GetObject(context.Background(), minIo.Vars["bucket"].(string), src, minio.GetObjectOptions{})
- if err != nil {
- return false, err
- }
- localFile, err := os.Create(target)
- if err != nil {
- return false, err
- }
- if _, err = io.Copy(localFile, object); err != nil {
- return false, err
- }
- return true, nil
- } else {
- return false, constant.ErrInvalidParams
- }
-}
-
-func (minIo *minIoClient) GetBucket() (string, error) {
- if _, ok := minIo.Vars["bucket"]; ok {
- return minIo.Vars["bucket"].(string), nil
- } else {
- return "", constant.ErrInvalidParams
- }
-}
-
-func (minIo minIoClient) ListObjects(prefix string) ([]string, error) {
- bucket, err := minIo.GetBucket()
+func (m minIoClient) Download(src, target string) (bool, error) {
+ object, err := m.client.GetObject(context.Background(), m.bucket, src, minio.GetObjectOptions{})
if err != nil {
- return nil, constant.ErrInvalidParams
+ return false, err
}
+ localFile, err := os.Create(target)
+ if err != nil {
+ return false, err
+ }
+ if _, err = io.Copy(localFile, object); err != nil {
+ return false, err
+ }
+ return true, nil
+}
+
+func (m minIoClient) ListObjects(prefix string) ([]string, error) {
opts := minio.ListObjectsOptions{
Recursive: true,
Prefix: prefix,
}
var result []string
- for object := range minIo.client.ListObjects(context.Background(), bucket, opts) {
+ for object := range m.client.ListObjects(context.Background(), m.bucket, opts) {
if object.Err != nil {
continue
}
diff --git a/backend/utils/cloud_storage/client/onedrive.go b/backend/utils/cloud_storage/client/onedrive.go
index f86d1e810..b349d9ff9 100644
--- a/backend/utils/cloud_storage/client/onedrive.go
+++ b/backend/utils/cloud_storage/client/onedrive.go
@@ -23,17 +23,11 @@ import (
)
type oneDriveClient struct {
- Vars map[string]interface{}
client odsdk.Client
}
func NewOneDriveClient(vars map[string]interface{}) (*oneDriveClient, error) {
- token := ""
- if _, ok := vars["accessToken"]; ok {
- token = vars["accessToken"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
+ token := loadParamFromVars("accessToken", true, vars)
ctx := context.Background()
newToken, err := refreshToken(token)
@@ -51,13 +45,13 @@ func NewOneDriveClient(vars map[string]interface{}) (*oneDriveClient, error) {
return &oneDriveClient{client: *client}, nil
}
-func (onedrive oneDriveClient) ListBuckets() ([]interface{}, error) {
+func (o oneDriveClient) ListBuckets() ([]interface{}, error) {
return nil, nil
}
-func (onedrive oneDriveClient) Exist(path string) (bool, error) {
+func (o oneDriveClient) Exist(path string) (bool, error) {
path = "/" + strings.TrimPrefix(path, "/")
- fileID, err := onedrive.loadIDByPath(path)
+ fileID, err := o.loadIDByPath(path)
if err != nil {
return false, err
}
@@ -65,26 +59,50 @@ func (onedrive oneDriveClient) Exist(path string) (bool, error) {
return len(fileID) != 0, nil
}
-func (onedrive oneDriveClient) Delete(path string) (bool, error) {
+func (o oneDriveClient) Size(path string) (int64, error) {
path = "/" + strings.TrimPrefix(path, "/")
- req, err := onedrive.client.NewRequest("DELETE", fmt.Sprintf("me/drive/root:%s", path), nil)
+ pathItem := "root:" + path
+ if path == "/" {
+ pathItem = "root"
+ }
+ req, err := o.client.NewRequest("GET", fmt.Sprintf("me/drive/%s", pathItem), nil)
+ if err != nil {
+ return 0, fmt.Errorf("new request for file id failed, err: %v", err)
+ }
+ var driveItem myDriverItem
+ if err := o.client.Do(context.Background(), req, false, &driveItem); err != nil {
+ return 0, fmt.Errorf("do request for file id failed, err: %v", err)
+ }
+
+ return driveItem.Size, nil
+}
+
+type myDriverItem struct {
+ Name string `json:"name"`
+ Id string `json:"id"`
+ Size int64 `json:"size"`
+}
+
+func (o oneDriveClient) Delete(path string) (bool, error) {
+ path = "/" + strings.TrimPrefix(path, "/")
+ req, err := o.client.NewRequest("DELETE", fmt.Sprintf("me/drive/root:%s", path), nil)
if err != nil {
return false, fmt.Errorf("new request for delete file failed, err: %v \n", err)
}
- if err := onedrive.client.Do(context.Background(), req, false, nil); err != nil {
+ if err := o.client.Do(context.Background(), req, false, nil); err != nil {
return false, fmt.Errorf("do request for delete file failed, err: %v \n", err)
}
return true, nil
}
-func (onedrive oneDriveClient) Upload(src, target string) (bool, error) {
+func (o oneDriveClient) Upload(src, target string) (bool, error) {
target = "/" + strings.TrimPrefix(target, "/")
- if _, err := onedrive.loadIDByPath(path.Dir(target)); err != nil {
+ if _, err := o.loadIDByPath(path.Dir(target)); err != nil {
if !strings.Contains(err.Error(), "itemNotFound") {
return false, err
}
- if err := onedrive.createFolder(path.Dir(target)); err != nil {
+ if err := o.createFolder(path.Dir(target)); err != nil {
return false, fmt.Errorf("create dir before upload failed, err: %v", err)
}
}
@@ -105,7 +123,7 @@ func (onedrive oneDriveClient) Upload(src, target string) (bool, error) {
fileName := fileInfo.Name()
fileSize := fileInfo.Size()
- folderID, err := onedrive.loadIDByPath(path.Dir(target))
+ folderID, err := o.loadIDByPath(path.Dir(target))
if err != nil {
return false, err
}
@@ -119,13 +137,13 @@ func (onedrive oneDriveClient) Upload(src, target string) (bool, error) {
DeferCommit bool `json:"deferCommit"`
}{sessionCreationRequestInside, false}
- sessionCreationReq, err := onedrive.client.NewRequest("POST", apiURL, sessionCreationRequest)
+ sessionCreationReq, err := o.client.NewRequest("POST", apiURL, sessionCreationRequest)
if err != nil {
return false, err
}
var sessionCreationResp *NewUploadSessionCreationResponse
- err = onedrive.client.Do(ctx, sessionCreationReq, false, &sessionCreationResp)
+ err = o.client.Do(ctx, sessionCreationReq, false, &sessionCreationResp)
if err != nil {
return false, fmt.Errorf("session creation failed %w", err)
}
@@ -149,11 +167,11 @@ func (onedrive oneDriveClient) Upload(src, target string) (bool, error) {
bufferLast := buffer[:length]
buffer = bufferLast
}
- sessionFileUploadReq, err := onedrive.NewSessionFileUploadRequest(fileSessionUploadUrl, splitNow*sizePerSplit, fileSize, bytes.NewReader(buffer))
+ sessionFileUploadReq, err := o.NewSessionFileUploadRequest(fileSessionUploadUrl, splitNow*sizePerSplit, fileSize, bytes.NewReader(buffer))
if err != nil {
return false, err
}
- if err := onedrive.client.Do(ctx, sessionFileUploadReq, false, &fileUploadResp); err != nil {
+ if err := o.client.Do(ctx, sessionFileUploadReq, false, &fileUploadResp); err != nil {
return false, err
}
}
@@ -164,14 +182,14 @@ func (onedrive oneDriveClient) Upload(src, target string) (bool, error) {
return true, nil
}
-func (onedrive oneDriveClient) Download(src, target string) (bool, error) {
+func (o oneDriveClient) Download(src, target string) (bool, error) {
src = "/" + strings.TrimPrefix(src, "/")
- req, err := onedrive.client.NewRequest("GET", fmt.Sprintf("me/drive/root:%s", src), nil)
+ req, err := o.client.NewRequest("GET", fmt.Sprintf("me/drive/root:%s", src), nil)
if err != nil {
return false, fmt.Errorf("new request for file id failed, err: %v", err)
}
var driveItem *odsdk.DriveItem
- if err := onedrive.client.Do(context.Background(), req, false, &driveItem); err != nil {
+ if err := o.client.Do(context.Background(), req, false, &driveItem); err != nil {
return false, fmt.Errorf("do request for file id failed, err: %v", err)
}
@@ -196,19 +214,19 @@ func (onedrive oneDriveClient) Download(src, target string) (bool, error) {
return true, nil
}
-func (onedrive *oneDriveClient) ListObjects(prefix string) ([]string, error) {
+func (o *oneDriveClient) ListObjects(prefix string) ([]string, error) {
prefix = "/" + strings.TrimPrefix(prefix, "/")
- folderID, err := onedrive.loadIDByPath(prefix)
+ folderID, err := o.loadIDByPath(prefix)
if err != nil {
return nil, err
}
- req, err := onedrive.client.NewRequest("GET", fmt.Sprintf("me/drive/items/%s/children", folderID), nil)
+ req, err := o.client.NewRequest("GET", fmt.Sprintf("me/drive/items/%s/children", folderID), nil)
if err != nil {
return nil, fmt.Errorf("new request for list failed, err: %v", err)
}
var driveItems *odsdk.OneDriveDriveItemsResponse
- if err := onedrive.client.Do(context.Background(), req, false, &driveItems); err != nil {
+ if err := o.client.Do(context.Background(), req, false, &driveItems); err != nil {
return nil, fmt.Errorf("do request for list failed, err: %v", err)
}
@@ -219,17 +237,17 @@ func (onedrive *oneDriveClient) ListObjects(prefix string) ([]string, error) {
return itemList, nil
}
-func (onedrive *oneDriveClient) loadIDByPath(path string) (string, error) {
+func (o *oneDriveClient) loadIDByPath(path string) (string, error) {
pathItem := "root:" + path
if path == "/" {
pathItem = "root"
}
- req, err := onedrive.client.NewRequest("GET", fmt.Sprintf("me/drive/%s", pathItem), nil)
+ req, err := o.client.NewRequest("GET", fmt.Sprintf("me/drive/%s", pathItem), nil)
if err != nil {
return "", fmt.Errorf("new request for file id failed, err: %v", err)
}
var driveItem *odsdk.DriveItem
- if err := onedrive.client.Do(context.Background(), req, false, &driveItem); err != nil {
+ if err := o.client.Do(context.Background(), req, false, &driveItem); err != nil {
return "", fmt.Errorf("do request for file id failed, err: %v", err)
}
return driveItem.Id, nil
@@ -269,18 +287,18 @@ func refreshToken(oldToken string) (string, error) {
return accessToken, nil
}
-func (onedrive *oneDriveClient) createFolder(parent string) error {
- if _, err := onedrive.loadIDByPath(path.Dir(parent)); err != nil {
+func (o *oneDriveClient) createFolder(parent string) error {
+ if _, err := o.loadIDByPath(path.Dir(parent)); err != nil {
if !strings.Contains(err.Error(), "itemNotFound") {
return err
}
- _ = onedrive.createFolder(path.Dir(parent))
+ _ = o.createFolder(path.Dir(parent))
}
- item2, err := onedrive.loadIDByPath(path.Dir(parent))
+ item2, err := o.loadIDByPath(path.Dir(parent))
if err != nil {
return err
}
- if _, err := onedrive.client.DriveItems.CreateNewFolder(context.Background(), "", item2, path.Base(parent)); err != nil {
+ if _, err := o.client.DriveItems.CreateNewFolder(context.Background(), "", item2, path.Base(parent)); err != nil {
return err
}
return nil
@@ -307,8 +325,8 @@ type DriveItem struct {
WebURL string `json:"webUrl"`
}
-func (onedrive *oneDriveClient) NewSessionFileUploadRequest(absoluteUrl string, grandOffset, grandTotalSize int64, byteReader *bytes.Reader) (*http.Request, error) {
- apiUrl, err := onedrive.client.BaseURL.Parse(absoluteUrl)
+func (o *oneDriveClient) NewSessionFileUploadRequest(absoluteUrl string, grandOffset, grandTotalSize int64, byteReader *bytes.Reader) (*http.Request, error) {
+ apiUrl, err := o.client.BaseURL.Parse(absoluteUrl)
if err != nil {
return nil, err
}
diff --git a/backend/utils/cloud_storage/client/oss.go b/backend/utils/cloud_storage/client/oss.go
index b9116e950..635039140 100644
--- a/backend/utils/cloud_storage/client/oss.go
+++ b/backend/utils/cloud_storage/client/oss.go
@@ -1,54 +1,36 @@
package client
import (
- "github.com/1Panel-dev/1Panel/backend/constant"
+ "fmt"
+
osssdk "github.com/aliyun/aliyun-oss-go-sdk/oss"
)
type ossClient struct {
- scType string
- Vars map[string]interface{}
- client osssdk.Client
+ scType string
+ bucketStr string
+ client osssdk.Client
}
func NewOssClient(vars map[string]interface{}) (*ossClient, error) {
- var endpoint string
- var accessKey string
- var secretKey string
- var scType string
- if _, ok := vars["endpoint"]; ok {
- endpoint = vars["endpoint"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["accessKey"]; ok {
- accessKey = vars["accessKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["scType"]; ok {
- scType = vars["scType"].(string)
- } else {
+ endpoint := loadParamFromVars("endpoint", true, vars)
+ accessKey := loadParamFromVars("accessKey", true, vars)
+ secretKey := loadParamFromVars("secretKey", true, vars)
+ bucketStr := loadParamFromVars("bucket", true, vars)
+ scType := loadParamFromVars("scType", true, vars)
+ if len(scType) == 0 {
scType = "Standard"
}
- if _, ok := vars["secretKey"]; ok {
- secretKey = vars["secretKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
client, err := osssdk.New(endpoint, accessKey, secretKey)
if err != nil {
return nil, err
}
- return &ossClient{
- scType: scType,
- Vars: vars,
- client: *client,
- }, nil
+
+ return &ossClient{scType: scType, bucketStr: bucketStr, client: *client}, nil
}
-func (oss ossClient) ListBuckets() ([]interface{}, error) {
- response, err := oss.client.ListBuckets()
+func (o ossClient) ListBuckets() ([]interface{}, error) {
+ response, err := o.client.ListBuckets()
if err != nil {
return nil, err
}
@@ -59,69 +41,72 @@ func (oss ossClient) ListBuckets() ([]interface{}, error) {
return result, err
}
-func (oss ossClient) Exist(path string) (bool, error) {
- bucket, err := oss.GetBucket()
+func (o ossClient) Exist(path string) (bool, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
if err != nil {
return false, err
}
return bucket.IsObjectExist(path)
-
}
-func (oss ossClient) Delete(path string) (bool, error) {
- bucket, err := oss.GetBucket()
+func (o ossClient) Size(path string) (int64, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
+ if err != nil {
+ return 0, err
+ }
+ lor, err := bucket.ListObjectsV2(osssdk.Prefix(path))
+ if err != nil {
+ return 0, err
+ }
+ if len(lor.Objects) == 0 {
+ return 0, fmt.Errorf("no such file %s", path)
+ }
+ return lor.Objects[0].Size, nil
+}
+
+func (o ossClient) Delete(path string) (bool, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
if err != nil {
return false, err
}
- err = bucket.DeleteObject(path)
- if err != nil {
+ if err := bucket.DeleteObject(path); err != nil {
return false, err
}
return true, nil
}
-func (oss ossClient) Upload(src, target string) (bool, error) {
- bucket, err := oss.GetBucket()
+func (o ossClient) Upload(src, target string) (bool, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
if err != nil {
return false, err
}
- err = bucket.UploadFile(target, src, 200*1024*1024, osssdk.Routines(5), osssdk.Checkpoint(true, ""), osssdk.ObjectStorageClass(osssdk.StorageClassType(oss.scType)))
- if err != nil {
+ if err := bucket.UploadFile(target, src,
+ 200*1024*1024,
+ osssdk.Routines(5),
+ osssdk.Checkpoint(true, ""),
+ osssdk.ObjectStorageClass(osssdk.StorageClassType(o.scType))); err != nil {
return false, err
}
return true, nil
}
-func (oss ossClient) Download(src, target string) (bool, error) {
- bucket, err := oss.GetBucket()
+func (o ossClient) Download(src, target string) (bool, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
if err != nil {
return false, err
}
- err = bucket.DownloadFile(src, target, 200*1024*1024, osssdk.Routines(5), osssdk.Checkpoint(true, ""))
- if err != nil {
+ if err := bucket.DownloadFile(src, target, 200*1024*1024, osssdk.Routines(5), osssdk.Checkpoint(true, "")); err != nil {
return false, err
}
return true, nil
}
-func (oss *ossClient) GetBucket() (*osssdk.Bucket, error) {
- if _, ok := oss.Vars["bucket"]; ok {
- bucket, err := oss.client.Bucket(oss.Vars["bucket"].(string))
- if err != nil {
- return nil, err
- }
- return bucket, nil
- } else {
- return nil, constant.ErrInvalidParams
- }
-}
-
-func (oss *ossClient) ListObjects(prefix string) ([]string, error) {
- bucket, err := oss.GetBucket()
+func (o *ossClient) ListObjects(prefix string) ([]string, error) {
+ bucket, err := o.client.Bucket(o.bucketStr)
if err != nil {
- return nil, constant.ErrInvalidParams
+ return nil, err
}
- lor, err := bucket.ListObjects(osssdk.Prefix(prefix))
+ lor, err := bucket.ListObjectsV2(osssdk.Prefix(prefix))
if err != nil {
return nil, err
}
diff --git a/backend/utils/cloud_storage/client/s3.go b/backend/utils/cloud_storage/client/s3.go
index a6bce842c..0f6cb96c8 100644
--- a/backend/utils/cloud_storage/client/s3.go
+++ b/backend/utils/cloud_storage/client/s3.go
@@ -3,7 +3,6 @@ package client
import (
"os"
- "github.com/1Panel-dev/1Panel/backend/constant"
"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/aws/awserr"
"github.com/aws/aws-sdk-go/aws/credentials"
@@ -14,41 +13,20 @@ import (
type s3Client struct {
scType string
- Vars map[string]interface{}
+ bucket string
Sess session.Session
}
func NewS3Client(vars map[string]interface{}) (*s3Client, error) {
- var accessKey string
- var secretKey string
- var endpoint string
- var scType string
- var region string
- if _, ok := vars["accessKey"]; ok {
- accessKey = vars["accessKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["secretKey"]; ok {
- secretKey = vars["secretKey"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["scType"]; ok {
- scType = vars["scType"].(string)
- } else {
+ accessKey := loadParamFromVars("accessKey", true, vars)
+ secretKey := loadParamFromVars("secretKey", true, vars)
+ endpoint := loadParamFromVars("endpoint", true, vars)
+ region := loadParamFromVars("region", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
+ scType := loadParamFromVars("scType", true, vars)
+ if len(scType) == 0 {
scType = "Standard"
}
- if _, ok := vars["endpoint"]; ok {
- endpoint = vars["endpoint"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["region"]; ok {
- region = vars["region"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
sess, err := session.NewSession(&aws.Config{
Credentials: credentials.NewStaticCredentials(accessKey, secretKey, ""),
Endpoint: aws.String(endpoint),
@@ -59,16 +37,12 @@ func NewS3Client(vars map[string]interface{}) (*s3Client, error) {
if err != nil {
return nil, err
}
- return &s3Client{
- scType: scType,
- Vars: vars,
- Sess: *sess,
- }, nil
+ return &s3Client{scType: scType, bucket: bucket, Sess: *sess}, nil
}
-func (s3C s3Client) ListBuckets() ([]interface{}, error) {
+func (s s3Client) ListBuckets() ([]interface{}, error) {
var result []interface{}
- svc := s3.New(&s3C.Sess)
+ svc := s3.New(&s.Sess)
res, err := svc.ListBuckets(nil)
if err != nil {
return nil, err
@@ -79,17 +53,12 @@ func (s3C s3Client) ListBuckets() ([]interface{}, error) {
return result, nil
}
-func (s3C s3Client) Exist(path string) (bool, error) {
- bucket, err := s3C.getBucket()
- if err != nil {
- return false, err
- }
- svc := s3.New(&s3C.Sess)
- _, err = svc.HeadObject(&s3.HeadObjectInput{
- Bucket: &bucket,
+func (s s3Client) Exist(path string) (bool, error) {
+ svc := s3.New(&s.Sess)
+ if _, err := svc.HeadObject(&s3.HeadObjectInput{
+ Bucket: &s.bucket,
Key: &path,
- })
- if err != nil {
+ }); err != nil {
if aerr, ok := err.(awserr.RequestFailure); ok {
if aerr.StatusCode() == 404 {
return false, nil
@@ -101,57 +70,53 @@ func (s3C s3Client) Exist(path string) (bool, error) {
return true, nil
}
-func (s3C s3Client) Delete(path string) (bool, error) {
- bucket, err := s3C.getBucket()
- if err != nil {
- return false, err
- }
- svc := s3.New(&s3C.Sess)
- _, err = svc.DeleteObject(&s3.DeleteObjectInput{Bucket: aws.String(bucket), Key: aws.String(path)})
- if err != nil {
- return false, err
- }
- err = svc.WaitUntilObjectNotExists(&s3.HeadObjectInput{
- Bucket: aws.String(bucket),
- Key: aws.String(path),
+func (s *s3Client) Size(path string) (int64, error) {
+ svc := s3.New(&s.Sess)
+ file, err := svc.GetObject(&s3.GetObjectInput{
+ Bucket: &s.bucket,
+ Key: &path,
})
if err != nil {
+ return 0, err
+ }
+ return *file.ContentLength, nil
+}
+
+func (s s3Client) Delete(path string) (bool, error) {
+ svc := s3.New(&s.Sess)
+ if _, err := svc.DeleteObject(&s3.DeleteObjectInput{Bucket: aws.String(s.bucket), Key: aws.String(path)}); err != nil {
+ return false, err
+ }
+ if err := svc.WaitUntilObjectNotExists(&s3.HeadObjectInput{
+ Bucket: aws.String(s.bucket),
+ Key: aws.String(path),
+ }); err != nil {
return false, err
}
return true, nil
}
-func (s3C s3Client) Upload(src, target string) (bool, error) {
- bucket, err := s3C.getBucket()
- if err != nil {
- return false, err
- }
+func (s s3Client) Upload(src, target string) (bool, error) {
file, err := os.Open(src)
if err != nil {
return false, err
}
defer file.Close()
- uploader := s3manager.NewUploader(&s3C.Sess)
- _, err = uploader.Upload(&s3manager.UploadInput{
- Bucket: aws.String(bucket),
+ uploader := s3manager.NewUploader(&s.Sess)
+ if _, err := uploader.Upload(&s3manager.UploadInput{
+ Bucket: aws.String(s.bucket),
Key: aws.String(target),
Body: file,
- StorageClass: &s3C.scType,
- })
- if err != nil {
+ StorageClass: &s.scType,
+ }); err != nil {
return false, err
}
return true, nil
}
-func (s3C s3Client) Download(src, target string) (bool, error) {
- bucket, err := s3C.getBucket()
- if err != nil {
- return false, err
- }
- _, err = os.Stat(target)
- if err != nil {
+func (s s3Client) Download(src, target string) (bool, error) {
+ if _, err := os.Stat(target); err != nil {
if os.IsNotExist(err) {
os.Remove(target)
} else {
@@ -163,44 +128,29 @@ func (s3C s3Client) Download(src, target string) (bool, error) {
return false, err
}
defer file.Close()
- downloader := s3manager.NewDownloader(&s3C.Sess)
- _, err = downloader.Download(file,
- &s3.GetObjectInput{
- Bucket: aws.String(bucket),
- Key: aws.String(src),
- })
- if err != nil {
+ downloader := s3manager.NewDownloader(&s.Sess)
+ if _, err = downloader.Download(file, &s3.GetObjectInput{
+ Bucket: aws.String(s.bucket),
+ Key: aws.String(src),
+ }); err != nil {
os.Remove(target)
return false, err
}
return true, nil
}
-func (s3C *s3Client) getBucket() (string, error) {
- if _, ok := s3C.Vars["bucket"]; ok {
- return s3C.Vars["bucket"].(string), nil
- } else {
- return "", constant.ErrInvalidParams
- }
-}
-
-func (s3C *s3Client) ListObjects(prefix string) ([]string, error) {
- bucket, err := s3C.getBucket()
- if err != nil {
- return nil, constant.ErrInvalidParams
- }
- svc := s3.New(&s3C.Sess)
+func (s *s3Client) ListObjects(prefix string) ([]string, error) {
+ svc := s3.New(&s.Sess)
var result []string
- if err := svc.ListObjectsPages(&s3.ListObjectsInput{
- Bucket: &bucket,
+ outputs, err := svc.ListObjects(&s3.ListObjectsInput{
+ Bucket: &s.bucket,
Prefix: &prefix,
- }, func(p *s3.ListObjectsOutput, last bool) (shouldContinue bool) {
- for _, obj := range p.Contents {
- result = append(result, *obj.Key)
- }
- return true
- }); err != nil {
- return nil, err
+ })
+ if err != nil {
+ return result, err
+ }
+ for _, item := range outputs.Contents {
+ result = append(result, *item.Key)
}
return result, nil
}
diff --git a/backend/utils/cloud_storage/client/sftp.go b/backend/utils/cloud_storage/client/sftp.go
index d8c53f8b3..63af2744a 100644
--- a/backend/utils/cloud_storage/client/sftp.go
+++ b/backend/utils/cloud_storage/client/sftp.go
@@ -6,53 +6,49 @@ import (
"net"
"os"
"path"
- "strconv"
"time"
- "github.com/1Panel-dev/1Panel/backend/constant"
"github.com/pkg/sftp"
"golang.org/x/crypto/ssh"
)
type sftpClient struct {
- Bucket string
- Vars map[string]interface{}
- client *sftp.Client
+ bucket string
+ connInfo string
+ config *ssh.ClientConfig
}
func NewSftpClient(vars map[string]interface{}) (*sftpClient, error) {
- if _, ok := vars["address"]; !ok {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["port"].(float64); !ok {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["password"]; !ok {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["username"]; !ok {
- return nil, constant.ErrInvalidParams
- }
- var bucket string
- if _, ok := vars["bucket"]; ok {
- bucket = vars["bucket"].(string)
- } else {
- return nil, constant.ErrInvalidParams
+ address := loadParamFromVars("address", true, vars)
+ port := loadParamFromVars("port", false, vars)
+ password := loadParamFromVars("password", true, vars)
+ username := loadParamFromVars("username", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
+
+ auth := []ssh.AuthMethod{ssh.Password(password)}
+ clientConfig := &ssh.ClientConfig{
+ User: username,
+ Auth: auth,
+ Timeout: 30 * time.Second,
+ HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error {
+ return nil
+ },
}
- port, err := strconv.Atoi(strconv.FormatFloat(vars["port"].(float64), 'G', -1, 64))
- if err != nil {
- return nil, err
- }
- sftpC, err := connect(vars["username"].(string), vars["password"].(string), vars["address"].(string), port)
- if err != nil {
- return nil, err
- }
- return &sftpClient{Bucket: bucket, client: sftpC, Vars: vars}, nil
+ return &sftpClient{bucket: bucket, connInfo: fmt.Sprintf("%s:%s", address, port), config: clientConfig}, nil
}
func (s sftpClient) Upload(src, target string) (bool, error) {
- defer s.client.Close()
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return false, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return false, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
srcFile, err := os.Open(src)
if err != nil {
@@ -60,18 +56,18 @@ func (s sftpClient) Upload(src, target string) (bool, error) {
}
defer srcFile.Close()
- targetFilePath := s.Bucket + "/" + target
+ targetFilePath := s.bucket + "/" + target
targetDir, _ := path.Split(targetFilePath)
- if _, err = s.client.Stat(targetDir); err != nil {
+ if _, err = client.Stat(targetDir); err != nil {
if os.IsNotExist(err) {
- if err = s.client.MkdirAll(targetDir); err != nil {
+ if err = client.MkdirAll(targetDir); err != nil {
return false, err
}
} else {
return false, err
}
}
- dstFile, err := s.client.Create(targetFilePath)
+ dstFile, err := client.Create(targetFilePath)
if err != nil {
return false, err
}
@@ -97,8 +93,17 @@ func (s sftpClient) ListBuckets() ([]interface{}, error) {
}
func (s sftpClient) Download(src, target string) (bool, error) {
- defer s.client.Close()
- srcFile, err := s.client.Open(s.Bucket + "/" + src)
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return false, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return false, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
+ srcFile, err := client.Open(s.bucket + "/" + src)
if err != nil {
return false, err
}
@@ -117,8 +122,18 @@ func (s sftpClient) Download(src, target string) (bool, error) {
}
func (s sftpClient) Exist(path string) (bool, error) {
- defer s.client.Close()
- srcFile, err := s.client.Open(s.Bucket + "/" + path)
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return false, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return false, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
+
+ srcFile, err := client.Open(s.bucket + "/" + path)
if err != nil {
if os.IsNotExist(err) {
return false, nil
@@ -130,48 +145,57 @@ func (s sftpClient) Exist(path string) (bool, error) {
return true, err
}
+func (s sftpClient) Size(path string) (int64, error) {
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return 0, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return 0, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
+
+ files, err := client.Stat(s.bucket + "/" + path)
+ if err != nil {
+ return 0, err
+ }
+ return files.Size(), nil
+}
+
func (s sftpClient) Delete(filePath string) (bool, error) {
- defer s.client.Close()
- targetFilePath := s.Bucket + "/" + filePath
- if err := s.client.Remove(targetFilePath); err != nil {
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return false, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return false, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
+
+ targetFilePath := s.bucket + "/" + filePath
+ if err := client.Remove(targetFilePath); err != nil {
return false, err
}
return true, nil
}
-func connect(user, password, host string, port int) (*sftp.Client, error) {
- var (
- auth []ssh.AuthMethod
- addr string
- clientConfig *ssh.ClientConfig
- sshClient *ssh.Client
- sftpClient *sftp.Client
- err error
- )
- auth = make([]ssh.AuthMethod, 0)
- auth = append(auth, ssh.Password(password))
- clientConfig = &ssh.ClientConfig{
- User: user,
- Auth: auth,
- Timeout: 30 * time.Second,
- HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error {
- return nil
- },
- }
- addr = fmt.Sprintf("%s:%d", host, port)
-
- if sshClient, err = ssh.Dial("tcp", addr, clientConfig); err != nil {
- return nil, err
- }
- if sftpClient, err = sftp.NewClient(sshClient); err != nil {
- return nil, err
- }
- return sftpClient, nil
-}
-
func (s sftpClient) ListObjects(prefix string) ([]string, error) {
- defer s.client.Close()
- files, err := s.client.ReadDir(s.Bucket + "/" + prefix)
+ sshClient, err := ssh.Dial("tcp", s.connInfo, s.config)
+ if err != nil {
+ return nil, err
+ }
+ client, err := sftp.NewClient(sshClient)
+ if err != nil {
+ return nil, err
+ }
+ defer client.Close()
+ defer sshClient.Close()
+
+ files, err := client.ReadDir(s.bucket + "/" + prefix)
if err != nil {
return nil, err
}
diff --git a/backend/utils/cloud_storage/client/webdav.go b/backend/utils/cloud_storage/client/webdav.go
index 1034631b3..56699ceeb 100644
--- a/backend/utils/cloud_storage/client/webdav.go
+++ b/backend/utils/cloud_storage/client/webdav.go
@@ -5,52 +5,26 @@ import (
"io"
"os"
- "github.com/1Panel-dev/1Panel/backend/constant"
"github.com/studio-b12/gowebdav"
)
type webDAVClient struct {
Bucket string
client *gowebdav.Client
- Vars map[string]interface{}
}
func NewWebDAVClient(vars map[string]interface{}) (*webDAVClient, error) {
- var (
- address string
- username string
- password string
- bucket string
- )
- if _, ok := vars["address"]; ok {
- address = vars["address"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["port"].(float64); !ok {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["username"]; ok {
- username = vars["username"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["password"]; ok {
- password = vars["password"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
- if _, ok := vars["bucket"]; ok {
- bucket = vars["bucket"].(string)
- } else {
- return nil, constant.ErrInvalidParams
- }
+ address := loadParamFromVars("address", true, vars)
+ port := loadParamFromVars("port", false, vars)
+ password := loadParamFromVars("password", true, vars)
+ username := loadParamFromVars("username", true, vars)
+ bucket := loadParamFromVars("bucket", true, vars)
- client := gowebdav.NewClient(fmt.Sprintf("%s:%v", address, vars["port"]), username, password)
+ client := gowebdav.NewClient(fmt.Sprintf("%s:%s", address, port), username, password)
if err := client.Connect(); err != nil {
return nil, err
}
- return &webDAVClient{Vars: vars, Bucket: bucket, client: client}, nil
+ return &webDAVClient{Bucket: bucket, client: client}, nil
}
func (s webDAVClient) Upload(src, target string) (bool, error) {
@@ -115,6 +89,14 @@ func (s webDAVClient) Exist(path string) (bool, error) {
return true, nil
}
+func (s webDAVClient) Size(path string) (int64, error) {
+ file, err := s.client.Stat(s.Bucket + "/" + path)
+ if err != nil {
+ return 0, err
+ }
+ return file.Size(), nil
+}
+
func (s webDAVClient) Delete(filePath string) (bool, error) {
if err := s.client.Remove(s.Bucket + "/" + filePath); err != nil {
return false, err
diff --git a/backend/utils/cloud_storage/cloud_storage_client.go b/backend/utils/cloud_storage/cloud_storage_client.go
index cf5745206..dbaea36e0 100644
--- a/backend/utils/cloud_storage/cloud_storage_client.go
+++ b/backend/utils/cloud_storage/cloud_storage_client.go
@@ -12,6 +12,8 @@ type CloudStorageClient interface {
Delete(path string) (bool, error)
Upload(src, target string) (bool, error)
Download(src, target string) (bool, error)
+
+ Size(path string) (int64, error)
}
func NewCloudStorageClient(backupType string, vars map[string]interface{}) (CloudStorageClient, error) {
diff --git a/frontend/src/components/backup/index.vue b/frontend/src/components/backup/index.vue
index 1d563b1da..73deceb1f 100644
--- a/frontend/src/components/backup/index.vue
+++ b/frontend/src/components/backup/index.vue
@@ -27,6 +27,14 @@
+
+
+
+ {{ computeSize(row.size) }}
+
+ -
+
+
@@ -52,7 +60,7 @@