This commit is contained in:
jiangyong27 2024-01-25 14:59:48 +08:00
parent bcfab4f3d8
commit b0877d1f86
5 changed files with 410 additions and 0 deletions

5
go.mod
View File

@ -7,6 +7,8 @@ require (
github.com/gin-gonic/gin v1.9.1
github.com/gomodule/redigo v1.8.9
github.com/influxdata/influxdb1-client v0.0.0-20220302092344-a9ab5670611c
github.com/minio/minio-go v6.0.14+incompatible
github.com/qiniu/go-sdk/v7 v7.19.0
github.com/sirupsen/logrus v1.9.3
github.com/smbrave/goutil v0.0.0-20240105134047-64fe0dfafba2
github.com/spf13/cast v1.5.0
@ -20,6 +22,7 @@ require (
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
github.com/gabriel-vasile/mimetype v1.4.2 // indirect
github.com/gin-contrib/sse v0.1.0 // indirect
github.com/go-ini/ini v1.67.0 // indirect
github.com/go-playground/locales v0.14.1 // indirect
github.com/go-playground/universal-translator v0.18.1 // indirect
github.com/go-playground/validator/v10 v10.14.0 // indirect
@ -30,6 +33,7 @@ require (
github.com/klauspost/cpuid/v2 v2.2.4 // indirect
github.com/leodido/go-urn v1.2.4 // indirect
github.com/mattn/go-isatty v0.0.19 // indirect
github.com/mitchellh/go-homedir v1.1.0 // indirect
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
github.com/modern-go/reflect2 v1.0.2 // indirect
github.com/pelletier/go-toml/v2 v2.0.8 // indirect
@ -37,6 +41,7 @@ require (
github.com/ugorji/go/codec v1.2.11 // indirect
golang.org/x/arch v0.3.0 // indirect
golang.org/x/net v0.10.0 // indirect
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4 // indirect
golang.org/x/sys v0.8.0 // indirect
golang.org/x/text v0.9.0 // indirect
google.golang.org/protobuf v1.30.0 // indirect

116
storage/minio.go Normal file
View File

@ -0,0 +1,116 @@
package storage
import (
"context"
"fmt"
"github.com/minio/minio-go"
"path"
"strings"
"time"
log "github.com/sirupsen/logrus"
)
type MinioConfig struct {
Bucket string
BaseUrl string
Endpoint string
AccessKey string
SecretKey string
Timeout time.Duration
}
type Minio struct {
inited bool
config *MinioConfig
client *minio.Client
}
func NewMinio(cfg *MinioConfig) Storage {
return &Minio{
config: cfg,
inited: false,
}
}
func (s *Minio) Init() error {
if s.inited {
return nil
}
c, err := minio.New(s.config.Endpoint, s.config.AccessKey, s.config.SecretKey, false)
if err != nil {
return err
}
s.client = c
s.inited = true
return nil
}
func (s *Minio) Put(fileName, objectName string, onProcess func(fsize, uploaded int64)) error {
if err := s.Init(); err != nil {
return err
}
ctx, cancel := context.WithTimeout(context.Background(), s.config.Timeout)
defer cancel()
objectName = strings.TrimLeft(objectName, "/ ")
ext := strings.TrimLeft(path.Ext(fileName), ".")
_, err := s.client.FPutObjectWithContext(ctx, s.config.Bucket, objectName, fileName,
minio.PutObjectOptions{ContentType: ext2ContentType(ext)})
if err != nil {
log.Errorf("upload file to minio error:%s", err.Error())
return err
}
return nil
}
func (s *Minio) Del(objectName string) error {
if err := s.Init(); err != nil {
return err
}
err := s.client.RemoveObject(s.config.Bucket, objectName)
if err != nil {
log.Errorf("delete file to minio error:%s", err.Error())
return err
}
return nil
}
func (s *Minio) Get(objectName, fileName string) error {
if err := s.Init(); err != nil {
return err
}
return s.client.FGetObject(s.config.Bucket, objectName, fileName, minio.GetObjectOptions{})
}
func (s *Minio) List(objectPrefix string) ([]string, error) {
if err := s.Init(); err != nil {
return nil, err
}
done := make(chan struct{})
defer close(done)
infos := s.client.ListObjectsV2(s.config.Bucket, objectPrefix, true, done)
result := make([]string, 0)
for info := range infos {
result = append(result, info.Key)
}
return result, nil
}
func (s *Minio) Url(objectName string, expire time.Duration) string {
return fmt.Sprintf("%s/%s/%s", s.config.BaseUrl, s.config.Bucket, objectName)
}
func (s *Minio) Stat(objectName string) (*ObjectInfo, error) {
return nil, nil
}
func (s *Minio) Fetch(url, objectName string) error {
return nil
}

213
storage/qiniu.go Normal file
View File

@ -0,0 +1,213 @@
package storage
import (
"context"
"errors"
"fmt"
"github.com/qiniu/go-sdk/v7/auth/qbox"
"github.com/qiniu/go-sdk/v7/storage"
"os"
"time"
)
type QiniuConfig struct {
Bucket string
AK string
SK string
Timeout time.Duration
}
type Qiniu struct {
config *QiniuConfig
}
func NewQiniu(cfg *QiniuConfig) Storage {
return &Qiniu{
config: cfg,
}
}
func (s *Qiniu) Put(fileName, objectName string, onProcess func(fsize, uploaded int64)) error {
mac := qbox.NewMac(s.config.AK, s.config.SK)
putPolicy := storage.PutPolicy{
Scope: s.config.Bucket,
Expires: uint64(time.Now().Add(12 * time.Hour).Unix()),
}
upToken := putPolicy.UploadToken(mac)
cfg := storage.Config{}
cfg.UseHTTPS = false
cfg.UseCdnDomains = false
// 构建表单上传的对象
formUploader := storage.NewFormUploader(&cfg)
ret := storage.PutRet{}
// 可选配置
putExtra := storage.PutExtra{
Params: map[string]string{},
OnProgress: onProcess,
}
err := formUploader.PutFile(context.Background(), &ret, upToken, objectName, fileName, &putExtra)
if err != nil {
return err
}
return nil
}
func (s *Qiniu) PutResume(fileName, objectName string, onProcess func(fsize, uploaded int64)) error {
mac := qbox.NewMac(s.config.AK, s.config.SK)
putPolicy := storage.PutPolicy{
Scope: s.config.Bucket,
}
upToken := putPolicy.UploadToken(mac)
cfg := storage.Config{}
cfg.UseHTTPS = false
cfg.UseCdnDomains = false
file, err := os.Open(fileName)
if err != nil {
return err
}
fileInfo, _ := file.Stat()
fileSize := fileInfo.Size()
resumeUploader := storage.NewResumeUploaderV2(&cfg)
ret := storage.PutRet{}
recorder, err := storage.NewFileRecorder(os.TempDir())
if err != nil {
return err
}
partSize := int64(1024 * 1024 * 1)
partUploaded := int64(0)
putExtra := storage.RputV2Extra{
Recorder: recorder,
Notify: func(partNumber int64, ret *storage.UploadPartsRet) {
partUploaded += 1
onProcess(fileSize, partUploaded*partSize)
},
PartSize: partSize,
}
err = resumeUploader.PutFile(context.Background(), &ret, upToken, objectName, fileName, &putExtra)
if err != nil {
return err
}
return nil
}
func (s *Qiniu) Del(objectName string) error {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(mac, &cfg)
return bucketManager.Delete(s.config.Bucket, objectName)
}
func (s *Qiniu) List(objectPrefix string) ([]string, error) {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
result := make([]string, 0)
marker := ""
bucketManager := storage.NewBucketManager(mac, &cfg)
for {
entries, _, nextMarker, hasNext, err := bucketManager.ListFiles(s.config.Bucket, objectPrefix, "", marker, 1000)
if err != nil {
return nil, err
}
for _, entry := range entries {
result = append(result, entry.Key)
}
if hasNext {
marker = nextMarker
} else {
break
}
}
return result, nil
}
func (s *Qiniu) Get(objectName, fileName string) error {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(mac, &cfg)
domains, err := bucketManager.ListBucketDomains(s.config.Bucket)
if err != nil {
return err
}
if len(domains) <= 0 {
return errors.New("bucket no domain")
}
url := fmt.Sprintf("http://%s/%s", domains[0].Domain, objectName)
return Download(url, fileName)
}
func (s *Qiniu) Url(objectName string, expire time.Duration) string {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(mac, &cfg)
domains, err := bucketManager.ListBucketDomains(s.config.Bucket)
if err != nil {
return ""
}
if len(domains) <= 0 {
return ""
}
domain := "http://" + domains[0].Domain
if expire == 0 {
return storage.MakePublicURLv2(domain, objectName)
} else {
deadline := time.Now().Add(expire).Unix()
return storage.MakePrivateURLv2(mac, domain, objectName, deadline)
}
}
func (s *Qiniu) Stat(objectName string) (*ObjectInfo, error) {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(mac, &cfg)
fileInfo, err := bucketManager.Stat(s.config.Bucket, objectName)
if err != nil {
return nil, err
}
info := new(ObjectInfo)
info.Size = fileInfo.Fsize
info.Hash = fileInfo.Hash
info.MimeType = fileInfo.MimeType
info.PutTime = fileInfo.PutTime
return info, nil
}
func (s *Qiniu) Fetch(url, objectName string) error {
mac := qbox.NewMac(s.config.AK, s.config.SK)
cfg := storage.Config{
UseHTTPS: false,
}
bucketManager := storage.NewBucketManager(mac, &cfg)
_, err := bucketManager.Fetch(url, s.config.Bucket, objectName)
if err != nil {
return err
}
return nil
}

20
storage/storage.go Normal file
View File

@ -0,0 +1,20 @@
package storage
import "time"
type ObjectInfo struct {
Size int64
MimeType string
Hash string
PutTime int64
}
type Storage interface {
Put(fileName, objectName string, onProcess func(fsize, uploaded int64)) error
Get(objectName, fileName string) error
Del(objectName string) error
Url(objectName string, expire time.Duration) string
Stat(objectName string) (*ObjectInfo, error)
List(objectPrefix string) ([]string, error)
Fetch(url, objectName string) error
}

56
storage/util.go Normal file
View File

@ -0,0 +1,56 @@
package storage
import (
"io"
"net/http"
"os"
"strconv"
"strings"
)
func contentType2Ext(contentType string) string {
if strings.Contains(contentType, "image/jpeg") {
return "jpg"
} else if strings.Contains(contentType, "audio/mpeg") {
return "mp3"
} else if strings.Contains(contentType, "video/mp4") {
return "mp4"
}
return ""
}
func ext2ContentType(ext string) string {
ext = strings.ToLower(ext)
if ext == "jpg" || ext == "jpeg" {
return "image/jpeg"
} else if ext == "png" {
return "image/png"
} else if ext == "mp3" {
return "audio/mpeg"
} else if ext == "mp4" {
return "video/mp4"
}
return ""
}
func Download(url, path string) error {
f, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666)
stat, err := f.Stat()
if err != nil {
return err
}
defer f.Close()
req, _ := http.NewRequest("GET", url, nil)
req.Header.Set("Range", "bytes="+strconv.FormatInt(stat.Size(), 10)+"-")
resp, err := http.DefaultClient.Do(req)
if err != nil {
return err
}
_, err = io.Copy(f, resp.Body)
if err != nil {
return err
}
return nil
}