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" } }