package sql import ( "context" "errors" "tercul/internal/domain" "time" "gorm.io/gorm" ) type passwordResetRepository struct { domain.BaseRepository[domain.PasswordReset] db *gorm.DB } // NewPasswordResetRepository creates a new PasswordResetRepository. func NewPasswordResetRepository(db *gorm.DB) domain.PasswordResetRepository { return &passwordResetRepository{ BaseRepository: NewBaseRepositoryImpl[domain.PasswordReset](db), db: db, } } // GetByToken finds a reset by token (only unused and non-expired) func (r *passwordResetRepository) GetByToken(ctx context.Context, token string) (*domain.PasswordReset, error) { var reset domain.PasswordReset if err := r.db.WithContext(ctx).Where("token = ? AND used = ? AND expires_at > ?", token, false, time.Now()).First(&reset).Error; err != nil { if errors.Is(err, gorm.ErrRecordNotFound) { return nil, ErrEntityNotFound } return nil, err } return &reset, nil } // GetByUserID finds resets by user ID func (r *passwordResetRepository) GetByUserID(ctx context.Context, userID uint) ([]domain.PasswordReset, error) { var resets []domain.PasswordReset if err := r.db.WithContext(ctx).Where("user_id = ?", userID).Find(&resets).Error; err != nil { return nil, err } return resets, nil } // DeleteExpired deletes expired resets func (r *passwordResetRepository) DeleteExpired(ctx context.Context) error { if err := r.db.WithContext(ctx).Where("expires_at < ?", time.Now()).Delete(&domain.PasswordReset{}).Error; err != nil { return err } return nil } // MarkAsUsed marks a reset as used func (r *passwordResetRepository) MarkAsUsed(ctx context.Context, id uint) error { if err := r.db.WithContext(ctx).Model(&domain.PasswordReset{}).Where("id = ?", id).Update("used", true).Error; err != nil { return err } return nil }