chore: remove direct avatar upload endpoint (POST /profile/avatar);
feat: add endpoints for presigned upload URLs (GET /upload/avatar, GET /upload/image); refactor: replace ProfileDto with NewProfileDto in update profile endpoint; feat: implement S3 integration for avatar management; fix: update database queries to handle new avatar upload flow; chore: add new dependencies for S3 handling (golang.org/x/time); refactor: rename UploadService to S3Service; refactor: change return type for func LocalizeS3Url(originalURL string) (*url.URL, error); feat: add custom validator for upload_id
This commit is contained in:
@@ -21,6 +21,7 @@ import (
|
||||
"easywish/internal/database"
|
||||
"easywish/internal/dto"
|
||||
errs "easywish/internal/errors"
|
||||
"easywish/internal/utils"
|
||||
mapspecial "easywish/internal/utils/mapSpecial"
|
||||
"errors"
|
||||
"time"
|
||||
@@ -36,10 +37,9 @@ import (
|
||||
type ProfileService interface {
|
||||
GetProfileByUsername(cinfo dto.ClientInfo, username string) (*dto.ProfileDto, error)
|
||||
GetMyProfile(cinfo dto.ClientInfo) (*dto.ProfileDto, error)
|
||||
UpdateProfile(cinfo dto.ClientInfo, newProfile dto.ProfileDto) (bool, error)
|
||||
UpdateProfile(cinfo dto.ClientInfo, newProfile dto.NewProfileDto) (bool, error)
|
||||
GetProfileSettings(cinfo dto.ClientInfo) (*dto.ProfileSettingsDto, error)
|
||||
UpdateProfileSettings(cinfo dto.ClientInfo, newProfileSettings dto.ProfileSettingsDto) (bool, error)
|
||||
UploadAvatar(cinfo dto.ClientInfo, filePath string) (*string, error)
|
||||
}
|
||||
|
||||
type profileServiceImpl struct {
|
||||
@@ -47,6 +47,7 @@ type profileServiceImpl struct {
|
||||
dbctx database.DbContext
|
||||
redis *redis.Client
|
||||
minio *minio.Client
|
||||
s3 S3Service
|
||||
}
|
||||
|
||||
func NewProfileService(_log *zap.Logger, _dbctx database.DbContext, _redis *redis.Client, _minio *minio.Client) ProfileService {
|
||||
@@ -117,12 +118,12 @@ func (p *profileServiceImpl) GetProfileSettings(cinfo dto.ClientInfo) (*dto.Prof
|
||||
}
|
||||
|
||||
// XXX: no validation for timestamps' allowed ranges
|
||||
func (p *profileServiceImpl) UpdateProfile(cinfo dto.ClientInfo, newProfile dto.ProfileDto) (bool, error) {
|
||||
func (p *profileServiceImpl) UpdateProfile(cinfo dto.ClientInfo, newProfile dto.NewProfileDto) (bool, error) {
|
||||
helper, db, err := database.NewDbHelperTransaction(p.dbctx); if err != nil {
|
||||
p.log.Error(
|
||||
"Failed to open transaction",
|
||||
zap.Error(err))
|
||||
return false, err
|
||||
return false, errs.ErrServerError
|
||||
}
|
||||
defer helper.Rollback()
|
||||
|
||||
@@ -131,11 +132,25 @@ func (p *profileServiceImpl) UpdateProfile(cinfo dto.ClientInfo, newProfile dto.
|
||||
Valid: true,
|
||||
}
|
||||
|
||||
var avatarUrl *string
|
||||
if newProfile.AvatarUploadID != nil {
|
||||
key, err := p.s3.SaveUpload(*newProfile.AvatarUploadID, "avatars"); if err != nil {
|
||||
p.log.Error("Failed to save avatar",
|
||||
zap.String("upload_id", *newProfile.AvatarUploadID),
|
||||
zap.Error(err))
|
||||
return false, errs.ErrServerError
|
||||
}
|
||||
|
||||
urlObj := p.s3.GetLocalizedFileUrl(*key, "avatars")
|
||||
avatarUrl = utils.NewPointer(urlObj.String())
|
||||
}
|
||||
|
||||
err = db.TXlessQueries.UpdateProfileByUsername(db.CTX, database.UpdateProfileByUsernameParams{
|
||||
Username: cinfo.Username,
|
||||
Name: newProfile.Name,
|
||||
Bio: newProfile.Bio,
|
||||
Birthday: birthdayTimestamp,
|
||||
AvatarUrl: avatarUrl,
|
||||
}); if err != nil {
|
||||
p.log.Error(
|
||||
"Failed to update user profile",
|
||||
@@ -193,8 +208,3 @@ func (p *profileServiceImpl) UpdateProfileSettings(cinfo dto.ClientInfo, newProf
|
||||
|
||||
return true, nil
|
||||
}
|
||||
|
||||
// TODO: implement S3 before I can do anything with it
|
||||
func (p *profileServiceImpl) UploadAvatar(cinfo dto.ClientInfo, filePath string) (*string, error) {
|
||||
panic("unimplemented")
|
||||
}
|
||||
|
||||
@@ -19,8 +19,11 @@ package services
|
||||
|
||||
import (
|
||||
"context"
|
||||
"easywish/config"
|
||||
minioclient "easywish/internal/minioClient"
|
||||
"easywish/internal/utils"
|
||||
"fmt"
|
||||
"net/url"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
@@ -28,21 +31,24 @@ import (
|
||||
"go.uber.org/zap"
|
||||
)
|
||||
|
||||
type UploadService interface {
|
||||
GetAvatarUrl() (*string, *map[string]string, error)
|
||||
GetImageUrl() (*string, *map[string]string, error)
|
||||
type S3Service interface {
|
||||
CreateAvatarUrl() (*string, *map[string]string, error)
|
||||
CreateImageUrl() (*string, *map[string]string, error)
|
||||
SaveUpload(uploadID string, bucket string) (*string, error)
|
||||
|
||||
GetLocalizedFileUrl(key string, bucket string) url.URL
|
||||
}
|
||||
|
||||
type uploadServiceImpl struct {
|
||||
type s3ServiceImpl struct {
|
||||
minio *minio.Client
|
||||
log *zap.Logger
|
||||
|
||||
avatarPolicy minio.PostPolicy
|
||||
imagePolicy minio.PostPolicy
|
||||
imagePolicy minio.PostPolicy
|
||||
}
|
||||
|
||||
func NewUploadService(_minio *minio.Client, _log *zap.Logger) UploadService {
|
||||
service := uploadServiceImpl{
|
||||
func NewUploadService(_minio *minio.Client, _log *zap.Logger) S3Service {
|
||||
service := s3ServiceImpl{
|
||||
minio: _minio,
|
||||
log: _log,
|
||||
}
|
||||
@@ -76,41 +82,88 @@ func NewUploadService(_minio *minio.Client, _log *zap.Logger) UploadService {
|
||||
return &service
|
||||
}
|
||||
|
||||
func (u *uploadServiceImpl) genUrl(policy minio.PostPolicy, prefix string) (*string, *map[string]string, error) {
|
||||
func (s *s3ServiceImpl) genUrl(policy minio.PostPolicy, prefix string) (*string, *map[string]string, error) {
|
||||
|
||||
object := prefix + uuid.New().String()
|
||||
|
||||
if err := policy.SetKey(object); err != nil {
|
||||
u.log.Error(
|
||||
s.log.Error(
|
||||
"Failed to set random key for presigned url",
|
||||
zap.Error(err))
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
url, formData, err := u.minio.PresignedPostPolicy(context.Background(), &policy)
|
||||
url, formData, err := s.minio.PresignedPostPolicy(context.Background(), &policy)
|
||||
if err != nil {
|
||||
u.log.Error(
|
||||
s.log.Error(
|
||||
"Failed to generate presigned url",
|
||||
zap.String("object", object),
|
||||
zap.Error(err))
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
convertedUrl, err := utils.LocalizeS3Url(url.String()); if err != nil {
|
||||
u.log.Error(
|
||||
convertedUrl, err := utils.LocalizeS3Url(url.String())
|
||||
if err != nil {
|
||||
s.log.Error(
|
||||
"Failed to localize object URL to user-accessible format",
|
||||
zap.String("url", url.String()),
|
||||
zap.Error(err))
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
return &convertedUrl, &formData, nil
|
||||
return utils.NewPointer(convertedUrl.String()), &formData, nil
|
||||
}
|
||||
|
||||
func (u *uploadServiceImpl) GetAvatarUrl() (*string, *map[string]string, error) {
|
||||
return u.genUrl(u.avatarPolicy, "avatar-")
|
||||
func (s *s3ServiceImpl) CreateAvatarUrl() (*string, *map[string]string, error) {
|
||||
return s.genUrl(s.avatarPolicy, "avatar-")
|
||||
}
|
||||
|
||||
func (u *uploadServiceImpl) GetImageUrl() (*string, *map[string]string, error) {
|
||||
return u.genUrl(u.imagePolicy, "image-")
|
||||
func (s *s3ServiceImpl) CreateImageUrl() (*string, *map[string]string, error) {
|
||||
return s.genUrl(s.imagePolicy, "image-")
|
||||
}
|
||||
|
||||
func (s *s3ServiceImpl) SaveUpload(uploadID string, bucketAlias string) (*string, error) {
|
||||
sourceBucket := minioclient.Buckets["uploads"]
|
||||
bucket := minioclient.Buckets[bucketAlias]
|
||||
newObjectKey := uuid.New().String()
|
||||
|
||||
_, err := s.minio.CopyObject(context.Background(), minio.CopyDestOptions{
|
||||
Bucket: bucket,
|
||||
Object: newObjectKey,
|
||||
}, minio.CopySrcOptions{
|
||||
Bucket: sourceBucket,
|
||||
Object: uploadID,
|
||||
})
|
||||
if err != nil {
|
||||
s.log.Error(
|
||||
"Failed to copy object to new bucket",
|
||||
zap.String("sourceBucket", sourceBucket),
|
||||
zap.String("uploadID", uploadID),
|
||||
zap.String("destinationBucket", bucket),
|
||||
zap.String("newObjectKey", newObjectKey),
|
||||
zap.Error(err))
|
||||
return nil, err
|
||||
}
|
||||
|
||||
err = s.minio.RemoveObject(context.Background(), sourceBucket, uploadID, minio.RemoveObjectOptions{})
|
||||
if err != nil {
|
||||
s.log.Error(
|
||||
"Failed to remove original object from uploads bucket",
|
||||
zap.String("sourceBucket", sourceBucket),
|
||||
zap.String("uploadID", uploadID),
|
||||
zap.Error(err))
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return &newObjectKey, nil
|
||||
}
|
||||
|
||||
func (s *s3ServiceImpl) GetLocalizedFileUrl(key string, bucketAlias string) url.URL {
|
||||
cfg := config.GetConfig()
|
||||
|
||||
return url.URL{
|
||||
Scheme: "http",
|
||||
Host: fmt.Sprintf("%s:%d", cfg.Hostname, cfg.Port),
|
||||
Path: fmt.Sprintf("/s3/%s/%s", minioclient.Buckets[bucketAlias], key),
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user