storeage
This commit is contained in:
parent
bcfab4f3d8
commit
b0877d1f86
5
go.mod
5
go.mod
|
@ -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
|
||||
|
|
|
@ -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
|
||||
}
|
|
@ -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
|
||||
}
|
|
@ -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
|
||||
}
|
|
@ -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
|
||||
}
|
Loading…
Reference in New Issue