171 lines
4.1 KiB
Go
171 lines
4.1 KiB
Go
|
|
package fileStorage
|
|||
|
|
|
|||
|
|
import (
|
|||
|
|
"context"
|
|||
|
|
"fmt"
|
|||
|
|
"io"
|
|||
|
|
"mime/multipart"
|
|||
|
|
"path/filepath"
|
|||
|
|
"strings"
|
|||
|
|
"time"
|
|||
|
|
|
|||
|
|
"backend/internal/svc"
|
|||
|
|
"backend/internal/types"
|
|||
|
|
|
|||
|
|
"github.com/google/uuid"
|
|||
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
type UploadVideoLogic struct {
|
|||
|
|
logx.Logger
|
|||
|
|
ctx context.Context
|
|||
|
|
svcCtx *svc.ServiceContext
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
func NewUploadVideoLogic(ctx context.Context, svcCtx *svc.ServiceContext) *UploadVideoLogic {
|
|||
|
|
return &UploadVideoLogic{
|
|||
|
|
Logger: logx.WithContext(ctx),
|
|||
|
|
ctx: ctx,
|
|||
|
|
svcCtx: svcCtx,
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
func (l *UploadVideoLogic) UploadVideo(req *types.UploadVideoReq, file multipart.File, header *multipart.FileHeader) (resp *types.UploadResp, err error) {
|
|||
|
|
// 驗證文件大小(100MB 限制)
|
|||
|
|
const maxVideoSize = 100 << 20 // 100MB
|
|||
|
|
if header.Size > maxVideoSize {
|
|||
|
|
return nil, fmt.Errorf("video size exceeds 100MB limit")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 讀取文件內容
|
|||
|
|
videoData, err := io.ReadAll(file)
|
|||
|
|
if err != nil {
|
|||
|
|
return nil, fmt.Errorf("failed to read file: %w", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 驗證實際文件大小
|
|||
|
|
if int64(len(videoData)) > maxVideoSize {
|
|||
|
|
return nil, fmt.Errorf("video size exceeds 100MB limit")
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 檢測視頻 MIME 類型
|
|||
|
|
mimeType := l.detectVideoType(videoData, header)
|
|||
|
|
if mimeType == "" {
|
|||
|
|
// 從文件名推斷
|
|||
|
|
ext := strings.ToLower(filepath.Ext(header.Filename))
|
|||
|
|
mimeType = l.getMimeTypeFromExtension(ext)
|
|||
|
|
if mimeType == "" {
|
|||
|
|
return nil, fmt.Errorf("unsupported video format")
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 生成唯一文件名
|
|||
|
|
fileExt := filepath.Ext(header.Filename)
|
|||
|
|
if fileExt == "" {
|
|||
|
|
fileExt = l.getExtensionFromMimeType(mimeType)
|
|||
|
|
}
|
|||
|
|
fileName := fmt.Sprintf("%s%s", uuid.New().String(), fileExt)
|
|||
|
|
objectPath := fmt.Sprintf("videos/%d/%s", time.Now().Year(), fileName)
|
|||
|
|
|
|||
|
|
// 上傳到 S3
|
|||
|
|
fileStorageUC := l.svcCtx.FileStorageUC
|
|||
|
|
if err := fileStorageUC.UploadFromData(l.ctx, videoData, objectPath, mimeType); err != nil {
|
|||
|
|
return nil, fmt.Errorf("failed to upload video: %w", err)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 獲取公開 URL
|
|||
|
|
fileUrl := fileStorageUC.GetPublicURL(l.ctx, objectPath)
|
|||
|
|
|
|||
|
|
return &types.UploadResp{
|
|||
|
|
FileUrl: fileUrl,
|
|||
|
|
FileSize: int64(len(videoData)),
|
|||
|
|
MimeType: mimeType,
|
|||
|
|
}, nil
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// detectVideoType 檢測視頻類型(通過 magic bytes)
|
|||
|
|
func (l *UploadVideoLogic) detectVideoType(data []byte, header *multipart.FileHeader) string {
|
|||
|
|
if len(data) < 12 {
|
|||
|
|
return ""
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 檢查常見視頻格式的 magic bytes
|
|||
|
|
// MP4: ftyp box 通常在開頭
|
|||
|
|
if len(data) >= 12 {
|
|||
|
|
// MP4/MOV: 通常以 ftyp 開頭
|
|||
|
|
if string(data[4:8]) == "ftyp" {
|
|||
|
|
if strings.Contains(string(data[8:12]), "mp4") || strings.Contains(string(data[8:12]), "isom") {
|
|||
|
|
return "video/mp4"
|
|||
|
|
}
|
|||
|
|
if strings.Contains(string(data[8:12]), "qt") {
|
|||
|
|
return "video/quicktime"
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// AVI: RIFF...AVI
|
|||
|
|
if len(data) >= 12 && string(data[0:4]) == "RIFF" && string(data[8:12]) == "AVI " {
|
|||
|
|
return "video/x-msvideo"
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// WebM: webm
|
|||
|
|
if len(data) >= 4 && string(data[0:4]) == "\x1a\x45\xdf\xa3" {
|
|||
|
|
return "video/webm"
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 從 Content-Type header 獲取
|
|||
|
|
if header != nil && len(header.Header) > 0 {
|
|||
|
|
contentType := header.Header.Get("Content-Type")
|
|||
|
|
if contentType != "" && strings.HasPrefix(contentType, "video/") {
|
|||
|
|
return contentType
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return ""
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// getMimeTypeFromExtension 根據文件擴展名獲取 MIME 類型
|
|||
|
|
func (l *UploadVideoLogic) getMimeTypeFromExtension(ext string) string {
|
|||
|
|
ext = strings.ToLower(ext)
|
|||
|
|
switch ext {
|
|||
|
|
case ".mp4":
|
|||
|
|
return "video/mp4"
|
|||
|
|
case ".mov":
|
|||
|
|
return "video/quicktime"
|
|||
|
|
case ".avi":
|
|||
|
|
return "video/x-msvideo"
|
|||
|
|
case ".webm":
|
|||
|
|
return "video/webm"
|
|||
|
|
case ".mkv":
|
|||
|
|
return "video/x-matroska"
|
|||
|
|
case ".flv":
|
|||
|
|
return "video/x-flv"
|
|||
|
|
case ".wmv":
|
|||
|
|
return "video/x-ms-wmv"
|
|||
|
|
default:
|
|||
|
|
return ""
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// getExtensionFromMimeType 根據 MIME 類型獲取文件擴展名
|
|||
|
|
func (l *UploadVideoLogic) getExtensionFromMimeType(mimeType string) string {
|
|||
|
|
switch mimeType {
|
|||
|
|
case "video/mp4":
|
|||
|
|
return ".mp4"
|
|||
|
|
case "video/quicktime":
|
|||
|
|
return ".mov"
|
|||
|
|
case "video/x-msvideo":
|
|||
|
|
return ".avi"
|
|||
|
|
case "video/webm":
|
|||
|
|
return ".webm"
|
|||
|
|
case "video/x-matroska":
|
|||
|
|
return ".mkv"
|
|||
|
|
case "video/x-flv":
|
|||
|
|
return ".flv"
|
|||
|
|
case "video/x-ms-wmv":
|
|||
|
|
return ".wmv"
|
|||
|
|
default:
|
|||
|
|
return ".mp4"
|
|||
|
|
}
|
|||
|
|
}
|