274 lines
8.5 KiB
Go
274 lines
8.5 KiB
Go
package storage
|
||
|
||
import (
|
||
"fmt"
|
||
"io"
|
||
"mime"
|
||
"mime/multipart"
|
||
"path/filepath"
|
||
"strings"
|
||
"time"
|
||
|
||
"dianshang/internal/config"
|
||
"dianshang/pkg/logger"
|
||
|
||
"github.com/aliyun/aliyun-oss-go-sdk/oss"
|
||
"github.com/google/uuid"
|
||
)
|
||
|
||
// OSSStorage 阿里云OSS存储服务
|
||
type OSSStorage struct {
|
||
client *oss.Client
|
||
bucket *oss.Bucket
|
||
config *config.OSSConfig
|
||
}
|
||
|
||
// NewOSSStorage 创建OSS存储服务实例
|
||
func NewOSSStorage(cfg *config.OSSConfig) (*OSSStorage, error) {
|
||
// 创建OSSClient实例
|
||
client, err := oss.New(cfg.Endpoint, cfg.AccessKeyID, cfg.AccessKeySecret)
|
||
if err != nil {
|
||
return nil, fmt.Errorf("创建OSS客户端失败: %w", err)
|
||
}
|
||
|
||
// 获取存储空间
|
||
bucket, err := client.Bucket(cfg.BucketName)
|
||
if err != nil {
|
||
return nil, fmt.Errorf("获取OSS Bucket失败: %w", err)
|
||
}
|
||
|
||
return &OSSStorage{
|
||
client: client,
|
||
bucket: bucket,
|
||
config: cfg,
|
||
}, nil
|
||
}
|
||
|
||
// UploadFile 上传文件到OSS
|
||
func (s *OSSStorage) UploadFile(file multipart.File, filename string, objectPath string) (string, error) {
|
||
// 生成OSS对象路径
|
||
objectKey := s.generateObjectKey(objectPath, filename)
|
||
logger.Infof("[OSS] 开始上传文件,对象路径: %s", objectKey)
|
||
|
||
// 获取文件MIME类型
|
||
contentType := getContentType(filename)
|
||
logger.Debugf("[OSS] 文件MIME类型: %s", contentType)
|
||
|
||
// 上传文件到OSS,设置Content-Type和其他元数据
|
||
// 使用 ObjectACL 设置为公共读,确保文件可以直接访问
|
||
err := s.bucket.PutObject(objectKey, file,
|
||
oss.ContentType(contentType),
|
||
oss.ObjectACL(oss.ACLPublicRead),
|
||
oss.ContentDisposition("inline"), // 关键:设置为inline而不是attachment
|
||
)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 上传文件失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("上传文件到OSS失败: %w", err)
|
||
}
|
||
|
||
// 获取OSS返回的文件URL
|
||
fileURL, err := s.GetFileURL(objectKey)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 获取文件URL失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("获取OSS文件URL失败: %w", err)
|
||
}
|
||
logger.Infof("[OSS] 上传成功,对象路径: %s, OSS返回URL: %s", objectKey, fileURL)
|
||
return fileURL, nil
|
||
}
|
||
|
||
// UploadFromBytes 从字节数组上传文件到OSS
|
||
func (s *OSSStorage) UploadFromBytes(data []byte, filename string, objectPath string) (string, error) {
|
||
// 生成OSS对象路径
|
||
objectKey := s.generateObjectKey(objectPath, filename)
|
||
logger.Infof("[OSS] 开始上传字节数组,对象路径: %s, 大小: %d bytes", objectKey, len(data))
|
||
|
||
// 获取文件MIME类型
|
||
contentType := getContentType(filename)
|
||
logger.Debugf("[OSS] 文件MIME类型: %s", contentType)
|
||
|
||
// 上传文件到OSS,设置Content-Type和其他元数据
|
||
err := s.bucket.PutObject(objectKey, strings.NewReader(string(data)),
|
||
oss.ContentType(contentType),
|
||
oss.ObjectACL(oss.ACLPublicRead),
|
||
oss.ContentDisposition("inline"),
|
||
)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 上传字节数组失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("上传文件到OSS失败: %w", err)
|
||
}
|
||
|
||
// 获取OSS返回的文件URL
|
||
fileURL, err := s.GetFileURL(objectKey)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 获取文件URL失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("获取OSS文件URL失败: %w", err)
|
||
}
|
||
logger.Infof("[OSS] 上传成功,对象路径: %s, OSS返回URL: %s", objectKey, fileURL)
|
||
return fileURL, nil
|
||
}
|
||
|
||
// UploadFromReader 从Reader上传文件到OSS
|
||
func (s *OSSStorage) UploadFromReader(reader io.Reader, filename string, objectPath string) (string, error) {
|
||
// 生成OSS对象路径
|
||
objectKey := s.generateObjectKey(objectPath, filename)
|
||
logger.Infof("[OSS] 开始上传Reader数据,对象路径: %s", objectKey)
|
||
|
||
// 获取文件MIME类型
|
||
contentType := getContentType(filename)
|
||
logger.Debugf("[OSS] 文件MIME类型: %s", contentType)
|
||
|
||
// 上传文件到OSS,设置Content-Type和其他元数据
|
||
err := s.bucket.PutObject(objectKey, reader,
|
||
oss.ContentType(contentType),
|
||
oss.ObjectACL(oss.ACLPublicRead),
|
||
oss.ContentDisposition("inline"),
|
||
)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 上传Reader数据失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("上传文件到OSS失败: %w", err)
|
||
}
|
||
|
||
// 获取OSS返回的文件URL
|
||
fileURL, err := s.GetFileURL(objectKey)
|
||
if err != nil {
|
||
logger.Errorf("[OSS] 获取文件URL失败,对象路径: %s, 错误: %v", objectKey, err)
|
||
return "", fmt.Errorf("获取OSS文件URL失败: %w", err)
|
||
}
|
||
logger.Infof("[OSS] 上传成功,对象路径: %s, OSS返回URL: %s", objectKey, fileURL)
|
||
return fileURL, nil
|
||
}
|
||
|
||
// DeleteFile 删除OSS中的文件
|
||
func (s *OSSStorage) DeleteFile(objectKey string) error {
|
||
err := s.bucket.DeleteObject(objectKey)
|
||
if err != nil {
|
||
return fmt.Errorf("删除OSS文件失败: %w", err)
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// IsObjectExist 检查对象是否存在
|
||
func (s *OSSStorage) IsObjectExist(objectKey string) (bool, error) {
|
||
exist, err := s.bucket.IsObjectExist(objectKey)
|
||
if err != nil {
|
||
return false, fmt.Errorf("检查OSS对象是否存在失败: %w", err)
|
||
}
|
||
return exist, nil
|
||
}
|
||
|
||
// GetFileURL 获取文件访问URL(使用OSS SDK标准方法)
|
||
func (s *OSSStorage) GetFileURL(objectKey string) (string, error) {
|
||
// 如果配置了自定义域名,使用自定义域名
|
||
if s.config.Domain != "" {
|
||
url := fmt.Sprintf("%s/%s", strings.TrimRight(s.config.Domain, "/"), objectKey)
|
||
logger.Debugf("[OSS] 使用自定义域名生成URL: %s", url)
|
||
return url, nil
|
||
}
|
||
|
||
// 使用OSS SDK获取对象的公共URL
|
||
// 对于公共读的Bucket,直接拼接URL
|
||
url := fmt.Sprintf("https://%s.%s/%s", s.config.BucketName, s.config.Endpoint, objectKey)
|
||
logger.Debugf("[OSS] 使用OSS标准域名生成URL: %s", url)
|
||
return url, nil
|
||
}
|
||
|
||
// GeneratePresignedURL 生成临时访问URL(带签名)
|
||
func (s *OSSStorage) GeneratePresignedURL(objectKey string, expireSeconds int64) (string, error) {
|
||
signedURL, err := s.bucket.SignURL(objectKey, oss.HTTPGet, expireSeconds)
|
||
if err != nil {
|
||
return "", fmt.Errorf("生成预签名URL失败: %w", err)
|
||
}
|
||
return signedURL, nil
|
||
}
|
||
|
||
// generateObjectKey 生成OSS对象键名
|
||
func (s *OSSStorage) generateObjectKey(objectPath string, filename string) string {
|
||
// 组合基础路径和对象路径
|
||
basePath := strings.TrimRight(s.config.BasePath, "/")
|
||
objectPath = strings.TrimLeft(objectPath, "/")
|
||
|
||
// OSS 路径统一使用斜杠,避免 Windows 下使用反斜杠
|
||
if basePath == "" {
|
||
return strings.ReplaceAll(filepath.ToSlash(filepath.Join(objectPath, filename)), "\\", "/")
|
||
}
|
||
|
||
return strings.ReplaceAll(filepath.ToSlash(filepath.Join(basePath, objectPath, filename)), "\\", "/")
|
||
}
|
||
|
||
// GenerateFilename 生成唯一文件名
|
||
func GenerateFilename(originalFilename string) string {
|
||
ext := filepath.Ext(originalFilename)
|
||
name := strings.TrimSuffix(originalFilename, ext)
|
||
|
||
// 生成时间戳和UUID
|
||
timestamp := time.Now().Format("20060102150405")
|
||
uuidStr := uuid.New().String()[:8]
|
||
|
||
// 清理文件名,移除特殊字符
|
||
name = strings.ReplaceAll(name, " ", "_")
|
||
name = strings.ReplaceAll(name, "(", "")
|
||
name = strings.ReplaceAll(name, ")", "")
|
||
|
||
return fmt.Sprintf("%s_%s_%s%s", name, timestamp, uuidStr, ext)
|
||
}
|
||
|
||
// IsValidImageType 验证图片文件类型
|
||
func IsValidImageType(filename string) bool {
|
||
ext := strings.ToLower(filepath.Ext(filename))
|
||
validExts := []string{".jpg", ".jpeg", ".png", ".gif", ".webp", ".svg"}
|
||
|
||
for _, validExt := range validExts {
|
||
if ext == validExt {
|
||
return true
|
||
}
|
||
}
|
||
return false
|
||
}
|
||
|
||
// getContentType 根据文件名获取MIME类型
|
||
func getContentType(filename string) string {
|
||
ext := strings.ToLower(filepath.Ext(filename))
|
||
|
||
// 常见图片类型
|
||
switch ext {
|
||
case ".jpg", ".jpeg":
|
||
return "image/jpeg"
|
||
case ".png":
|
||
return "image/png"
|
||
case ".gif":
|
||
return "image/gif"
|
||
case ".webp":
|
||
return "image/webp"
|
||
case ".svg":
|
||
return "image/svg+xml"
|
||
case ".bmp":
|
||
return "image/bmp"
|
||
case ".ico":
|
||
return "image/x-icon"
|
||
// 常见文档类型
|
||
case ".pdf":
|
||
return "application/pdf"
|
||
case ".doc":
|
||
return "application/msword"
|
||
case ".docx":
|
||
return "application/vnd.openxmlformats-officedocument.wordprocessingml.document"
|
||
case ".xls":
|
||
return "application/vnd.ms-excel"
|
||
case ".xlsx":
|
||
return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"
|
||
case ".txt":
|
||
return "text/plain; charset=utf-8"
|
||
case ".json":
|
||
return "application/json"
|
||
case ".xml":
|
||
return "application/xml"
|
||
default:
|
||
// 使用Go标准库自动检测
|
||
if mimeType := mime.TypeByExtension(ext); mimeType != "" {
|
||
return mimeType
|
||
}
|
||
// 默认类型
|
||
return "application/octet-stream"
|
||
}
|
||
}
|