UPDATE: Media module
All checks were successful
Build and Release / release (push) Successful in 1m7s

This commit is contained in:
2026-04-05 22:25:43 +07:00
parent eb404b37e9
commit 2d36004ac7
24 changed files with 1972 additions and 94 deletions

View File

@@ -1,6 +1,8 @@
package constants
const (
StreamEmailName = "stream:email_tasks"
GroupEmailName = "email_workers_group"
StreamEmailName = "stream:email_tasks"
StreamStorageName = "stream:storage_tasks"
GroupEmailName = "email_workers_group"
GroupStorageName = "storage_workers_group"
)

View File

@@ -4,8 +4,9 @@ type TaskType string
const (
TaskTypeSendEmailOTP TaskType = "SEND_EMAIL_OTP"
)
TaskTypeDeleteMedia TaskType = "DELETE_MEDIA"
)
func (t TaskType) String() string {
return string(t)
}
}

View File

@@ -10,4 +10,6 @@ const (
AccessTokenDuration = 15 * time.Minute
RefreshTokenDuration = 7 * 24 * time.Hour
TokenVerifiedDuration = 10 * time.Minute
PreSignedURLDuration = 15 * time.Minute
TokenUploadDuration = 1 * time.Hour
)

View File

@@ -6,7 +6,7 @@ const (
TokenPasswordReset TokenType = 1
TokenEmailVerify TokenType = 2
TokenMagicLink TokenType = 3
TokenRefreshToken TokenType = 4
TokenUpload TokenType = 4
)
func (t TokenType) String() string {
@@ -17,8 +17,8 @@ func (t TokenType) String() string {
return "EMAIL_VERIFY"
case TokenMagicLink:
return "LOGIN_MAGIC_LINK"
case TokenRefreshToken:
return "REFRESH_TOKEN"
case TokenUpload:
return "UPLOAD"
default:
return "UNKNOWN"
}
@@ -37,7 +37,7 @@ func ParseTokenType(v int16) TokenType {
case 3:
return TokenMagicLink
case 4:
return TokenRefreshToken
return TokenUpload
default:
return 0
}
@@ -51,9 +51,9 @@ func ParseTokenTypeFromString(s string) TokenType {
return TokenEmailVerify
case "LOGIN_MAGIC_LINK":
return TokenMagicLink
case "REFRESH_TOKEN":
return TokenRefreshToken
case "UPLOAD":
return TokenUpload
default:
return 0
}
}
}

View File

@@ -2,7 +2,9 @@ package storage
import (
"context"
"fmt"
"io"
"net/url"
"time"
"github.com/aws/aws-sdk-go-v2/aws"
@@ -20,18 +22,27 @@ type UploadOptions struct {
Metadata map[string]string
}
type MoveOptions struct {
Bucket string
Key string
}
type Storage interface {
Upload(ctx context.Context, key string, body io.Reader, size int64, opts UploadOptions) error
Move(ctx context.Context, src *MoveOptions, dest *MoveOptions) error
PresignUpload(ctx context.Context, key string, expire time.Duration, opts UploadOptions) (string, error)
GetURL(ctx context.Context, key string, expire time.Duration) (string, error)
Delete(ctx context.Context, key string) error
GetMainBucket() string
GetTempBucket() string
}
type s3Storage struct {
client *s3.Client
ps *s3.PresignClient
bucket string
endPoint string
client *s3.Client
ps *s3.PresignClient
bucket string
tempBucket string
endPoint string
}
func NewS3Storage() (Storage, error) {
@@ -49,16 +60,26 @@ func NewS3Storage() (Storage, error) {
if err != nil {
return nil, err
}
tempBucketName, err := ffconfig.GetConfig("STORAGE_BUCKET_TEMP_NAME")
if err != nil {
return nil, err
}
endpoint, err := ffconfig.GetConfig("STORAGE_ENDPOINT")
if err != nil {
return nil, err
}
region, err := ffconfig.GetConfig("STORAGE_REGION")
if err != nil {
return nil, err
}
cfg, err := config.LoadDefaultConfig(context.TODO(),
config.WithCredentialsProvider(credentials.NewStaticCredentialsProvider(accessKey, secretAccessKey, "")),
config.WithRegion("auto"),
config.WithRegion(region),
)
if err != nil {
log.Error().Msgf("unable to load AWS SDK config, %v", err)
return nil, err
@@ -66,15 +87,42 @@ func NewS3Storage() (Storage, error) {
client := s3.NewFromConfig(cfg, func(o *s3.Options) {
o.BaseEndpoint = aws.String(endpoint)
o.UsePathStyle = true
})
return &s3Storage{
client: client,
ps: s3.NewPresignClient(client),
bucket: bucketName,
endPoint: endpoint,
client: client,
ps: s3.NewPresignClient(client),
bucket: bucketName,
tempBucket: tempBucketName,
endPoint: endpoint,
}, nil
}
func (s *s3Storage) GetMainBucket() string { return s.bucket }
func (s *s3Storage) GetTempBucket() string { return s.tempBucket }
func (s *s3Storage) Move(ctx context.Context, src *MoveOptions, dest *MoveOptions) error {
copySource := url.PathEscape(fmt.Sprintf("%s/%s", src.Bucket, src.Key))
_, err := s.client.CopyObject(ctx, &s3.CopyObjectInput{
Bucket: aws.String(dest.Bucket),
Key: aws.String(dest.Key),
CopySource: aws.String(copySource),
})
if err != nil {
return fmt.Errorf("failed to copy object: %w", err)
}
_, err = s.client.DeleteObject(ctx, &s3.DeleteObjectInput{
Bucket: aws.String(src.Bucket),
Key: aws.String(src.Key),
})
if err != nil {
log.Error().Err(err).Msg("failed to delete source object after copy")
}
return nil
}
func (s *s3Storage) Upload(ctx context.Context, key string, body io.Reader, size int64, opts UploadOptions) error {
input := &s3.PutObjectInput{
@@ -99,7 +147,7 @@ func (s *s3Storage) Upload(ctx context.Context, key string, body io.Reader, size
func (s *s3Storage) PresignUpload(ctx context.Context, key string, expire time.Duration, opts UploadOptions) (string, error) {
input := &s3.PutObjectInput{
Bucket: &s.bucket,
Bucket: &s.tempBucket,
Key: &key,
}
@@ -119,6 +167,7 @@ func (s *s3Storage) PresignUpload(ctx context.Context, key string, expire time.D
}
return req.URL, nil
}
func (s *s3Storage) GetURL(ctx context.Context, key string, expire time.Duration) (string, error) {
req, err := s.ps.PresignGetObject(ctx, &s3.GetObjectInput{
Bucket: &s.bucket,