ultisuite-backend/internal/users/avatar.go
R3D347HR4Y 1d063237b9
Some checks are pending
CI / Go tests (push) Waiting to run
CI / Integration tests (push) Waiting to run
CI / DB migrations (push) Waiting to run
feat(transcription): integrate Faster Whisper for Jitsi transcriptions
- Added support for Faster Whisper transcription via Jigasi and Skynet.
- Updated .env.example to include new environment variables for transcription settings.
- Enhanced Jitsi Docker Compose configuration to include Skynet and Jigasi services.
- Introduced new API endpoints for managing organizational folders in the drive service.
- Updated Nextcloud initialization script to enable external file mounting.
- Improved error handling and response structures in the drive API.
- Added new properties for organization settings related to transcription and agenda management.
2026-06-12 19:10:18 +02:00

135 lines
3.2 KiB
Go

package users
import (
"context"
"encoding/base64"
"errors"
"fmt"
"strings"
"github.com/jackc/pgx/v5"
"github.com/jackc/pgx/v5/pgxpool"
)
const maxAvatarBytes = 512 * 1024
var (
ErrAvatarTooLarge = errors.New("avatar too large")
ErrAvatarInvalid = errors.New("avatar invalid")
ErrAvatarNotFound = errors.New("avatar not found")
)
var allowedAvatarMIME = map[string]struct{}{
"image/jpeg": {},
"image/png": {},
"image/gif": {},
"image/webp": {},
}
// GetAvatarURL returns the stored avatar URL/data URI for external_id.
func GetAvatarURL(ctx context.Context, db *pgxpool.Pool, externalID string) (string, error) {
if db == nil || strings.TrimSpace(externalID) == "" {
return "", nil
}
var avatarURL *string
err := db.QueryRow(ctx, `
SELECT avatar_url FROM users WHERE external_id = $1
`, externalID).Scan(&avatarURL)
if errors.Is(err, pgx.ErrNoRows) {
return "", nil
}
if err != nil {
return "", err
}
if avatarURL == nil {
return "", nil
}
return strings.TrimSpace(*avatarURL), nil
}
// SetAvatarURL validates and stores avatar_url for external_id.
func SetAvatarURL(ctx context.Context, db *pgxpool.Pool, externalID, avatarURL string) error {
if db == nil {
return fmt.Errorf("database not configured")
}
if strings.TrimSpace(externalID) == "" {
return fmt.Errorf("missing external id")
}
normalized, err := normalizeAvatarURL(avatarURL)
if err != nil {
return err
}
tag, err := db.Exec(ctx, `
UPDATE users
SET avatar_url = $2, updated_at = NOW()
WHERE external_id = $1
`, externalID, normalized)
if err != nil {
return err
}
if tag.RowsAffected() == 0 {
return pgx.ErrNoRows
}
return nil
}
// ClearAvatarURL removes the stored avatar for external_id.
func ClearAvatarURL(ctx context.Context, db *pgxpool.Pool, externalID string) error {
if db == nil {
return fmt.Errorf("database not configured")
}
if strings.TrimSpace(externalID) == "" {
return fmt.Errorf("missing external id")
}
tag, err := db.Exec(ctx, `
UPDATE users
SET avatar_url = NULL, updated_at = NOW()
WHERE external_id = $1
`, externalID)
if err != nil {
return err
}
if tag.RowsAffected() == 0 {
return pgx.ErrNoRows
}
return nil
}
func normalizeAvatarURL(raw string) (string, error) {
trimmed := strings.TrimSpace(raw)
if trimmed == "" {
return "", ErrAvatarInvalid
}
if strings.HasPrefix(trimmed, "https://") || strings.HasPrefix(trimmed, "http://") {
if len(trimmed) > 2048 {
return "", ErrAvatarTooLarge
}
return trimmed, nil
}
if !strings.HasPrefix(trimmed, "data:") {
return "", ErrAvatarInvalid
}
comma := strings.Index(trimmed, ",")
if comma == -1 {
return "", ErrAvatarInvalid
}
meta := trimmed[:comma]
payload := strings.TrimSpace(trimmed[comma+1:])
if !strings.Contains(meta, ";base64") {
return "", ErrAvatarInvalid
}
mimePart := strings.TrimPrefix(meta, "data:")
mimePart = strings.Split(mimePart, ";")[0]
if _, ok := allowedAvatarMIME[strings.ToLower(mimePart)]; !ok {
return "", ErrAvatarInvalid
}
decoded, err := base64.StdEncoding.DecodeString(payload)
if err != nil {
return "", ErrAvatarInvalid
}
if len(decoded) == 0 || len(decoded) > maxAvatarBytes {
return "", ErrAvatarTooLarge
}
return trimmed, nil
}