- Fix websearch provider failover: proxy error from provider-specific proxy now continues to next provider instead of aborting the entire loop - Fix SMTP failure locking users out: send email first, then write cache and increment rate counter - Fix notify email cache key case sensitivity: normalize to lowercase - Add OriginalPrice validation to validatePlanPatch and validatePlanRequired - Add empty scope validation for channel pricing rules (group_ids/account_ids) - Add platform color to account search dropdown in channel pricing rules
177 lines
5.1 KiB
Go
177 lines
5.1 KiB
Go
package repository
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
|
"github.com/redis/go-redis/v9"
|
|
)
|
|
|
|
const (
|
|
verifyCodeKeyPrefix = "verify_code:"
|
|
notifyVerifyKeyPrefix = "notify_verify:"
|
|
passwordResetKeyPrefix = "password_reset:"
|
|
passwordResetSentAtKeyPrefix = "password_reset_sent:"
|
|
notifyCodeUserRateKeyPrefix = "notify_code_user_rate:"
|
|
)
|
|
|
|
// verifyCodeKey generates the Redis key for email verification code.
|
|
func verifyCodeKey(email string) string {
|
|
return verifyCodeKeyPrefix + email
|
|
}
|
|
|
|
// notifyVerifyKey generates the Redis key for notify email verification code.
|
|
// Email is lowercased to prevent case-sensitive key mismatch (the business layer
|
|
// uses strings.EqualFold for comparison).
|
|
func notifyVerifyKey(email string) string {
|
|
return notifyVerifyKeyPrefix + strings.ToLower(email)
|
|
}
|
|
|
|
// passwordResetKey generates the Redis key for password reset token.
|
|
func passwordResetKey(email string) string {
|
|
return passwordResetKeyPrefix + email
|
|
}
|
|
|
|
// passwordResetSentAtKey generates the Redis key for password reset email sent timestamp.
|
|
func passwordResetSentAtKey(email string) string {
|
|
return passwordResetSentAtKeyPrefix + email
|
|
}
|
|
|
|
type emailCache struct {
|
|
rdb *redis.Client
|
|
}
|
|
|
|
func NewEmailCache(rdb *redis.Client) service.EmailCache {
|
|
return &emailCache{rdb: rdb}
|
|
}
|
|
|
|
func (c *emailCache) GetVerificationCode(ctx context.Context, email string) (*service.VerificationCodeData, error) {
|
|
key := verifyCodeKey(email)
|
|
val, err := c.rdb.Get(ctx, key).Result()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var data service.VerificationCodeData
|
|
if err := json.Unmarshal([]byte(val), &data); err != nil {
|
|
return nil, err
|
|
}
|
|
return &data, nil
|
|
}
|
|
|
|
func (c *emailCache) SetVerificationCode(ctx context.Context, email string, data *service.VerificationCodeData, ttl time.Duration) error {
|
|
key := verifyCodeKey(email)
|
|
val, err := json.Marshal(data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.rdb.Set(ctx, key, val, ttl).Err()
|
|
}
|
|
|
|
func (c *emailCache) DeleteVerificationCode(ctx context.Context, email string) error {
|
|
key := verifyCodeKey(email)
|
|
return c.rdb.Del(ctx, key).Err()
|
|
}
|
|
|
|
// Password reset token methods
|
|
|
|
func (c *emailCache) GetPasswordResetToken(ctx context.Context, email string) (*service.PasswordResetTokenData, error) {
|
|
key := passwordResetKey(email)
|
|
val, err := c.rdb.Get(ctx, key).Result()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var data service.PasswordResetTokenData
|
|
if err := json.Unmarshal([]byte(val), &data); err != nil {
|
|
return nil, err
|
|
}
|
|
return &data, nil
|
|
}
|
|
|
|
func (c *emailCache) SetPasswordResetToken(ctx context.Context, email string, data *service.PasswordResetTokenData, ttl time.Duration) error {
|
|
key := passwordResetKey(email)
|
|
val, err := json.Marshal(data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.rdb.Set(ctx, key, val, ttl).Err()
|
|
}
|
|
|
|
func (c *emailCache) DeletePasswordResetToken(ctx context.Context, email string) error {
|
|
key := passwordResetKey(email)
|
|
return c.rdb.Del(ctx, key).Err()
|
|
}
|
|
|
|
// Password reset email cooldown methods
|
|
|
|
func (c *emailCache) IsPasswordResetEmailInCooldown(ctx context.Context, email string) bool {
|
|
key := passwordResetSentAtKey(email)
|
|
exists, err := c.rdb.Exists(ctx, key).Result()
|
|
return err == nil && exists > 0
|
|
}
|
|
|
|
func (c *emailCache) SetPasswordResetEmailCooldown(ctx context.Context, email string, ttl time.Duration) error {
|
|
key := passwordResetSentAtKey(email)
|
|
return c.rdb.Set(ctx, key, "1", ttl).Err()
|
|
}
|
|
|
|
// Notify email verification code methods
|
|
|
|
func (c *emailCache) GetNotifyVerifyCode(ctx context.Context, email string) (*service.VerificationCodeData, error) {
|
|
key := notifyVerifyKey(email)
|
|
val, err := c.rdb.Get(ctx, key).Result()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var data service.VerificationCodeData
|
|
if err := json.Unmarshal([]byte(val), &data); err != nil {
|
|
return nil, err
|
|
}
|
|
return &data, nil
|
|
}
|
|
|
|
func (c *emailCache) SetNotifyVerifyCode(ctx context.Context, email string, data *service.VerificationCodeData, ttl time.Duration) error {
|
|
key := notifyVerifyKey(email)
|
|
val, err := json.Marshal(data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.rdb.Set(ctx, key, val, ttl).Err()
|
|
}
|
|
|
|
func (c *emailCache) DeleteNotifyVerifyCode(ctx context.Context, email string) error {
|
|
key := notifyVerifyKey(email)
|
|
return c.rdb.Del(ctx, key).Err()
|
|
}
|
|
|
|
// User-level rate limiting for notify email verification codes
|
|
|
|
func notifyCodeUserRateKey(userID int64) string {
|
|
return notifyCodeUserRateKeyPrefix + fmt.Sprintf("%d", userID)
|
|
}
|
|
|
|
func (c *emailCache) IncrNotifyCodeUserRate(ctx context.Context, userID int64, window time.Duration) (int64, error) {
|
|
key := notifyCodeUserRateKey(userID)
|
|
count, err := c.rdb.Incr(ctx, key).Result()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
// Always set TTL (idempotent) to avoid orphan keys if process crashes between INCR and EXPIRE.
|
|
if err := c.rdb.Expire(ctx, key, window).Err(); err != nil {
|
|
return count, fmt.Errorf("expire notify code rate key: %w", err)
|
|
}
|
|
return count, nil
|
|
}
|
|
|
|
func (c *emailCache) GetNotifyCodeUserRate(ctx context.Context, userID int64) (int64, error) {
|
|
key := notifyCodeUserRateKey(userID)
|
|
count, err := c.rdb.Get(ctx, key).Int64()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return count, nil
|
|
}
|