veza/veza-backend-api/internal/services/track_recommendation_service_test.go

264 lines
6 KiB
Go

package services
import (
"testing"
"time"
"github.com/google/uuid"
"go.uber.org/zap"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
"veza-backend-api/internal/models"
)
func TestNewTrackRecommendationService(t *testing.T) {
db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
if err != nil {
t.Fatalf("Failed to open database: %v", err)
}
logger := zap.NewNop()
service := NewTrackRecommendationService(db, logger)
if service == nil {
t.Error("NewTrackRecommendationService() returned nil")
}
if service.db == nil {
t.Error("NewTrackRecommendationService() returned service with nil db")
}
if service.logger == nil {
t.Error("NewTrackRecommendationService() returned service with nil logger")
}
}
func TestTrackRecommendationService_calculatePopularityScore(t *testing.T) {
db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
if err != nil {
t.Fatalf("Failed to open database: %v", err)
}
service := NewTrackRecommendationService(db, zap.NewNop())
tests := []struct {
name string
track *models.Track
wantMin float64
wantMax float64
}{
{
name: "high popularity track",
track: &models.Track{
PlayCount: 10000,
LikeCount: 1000,
},
wantMin: 0.8,
wantMax: 1.0,
},
{
name: "medium popularity track",
track: &models.Track{
PlayCount: 1000,
LikeCount: 100,
},
wantMin: 0.3,
wantMax: 0.8,
},
{
name: "low popularity track",
track: &models.Track{
PlayCount: 10,
LikeCount: 1,
},
wantMin: 0.0,
wantMax: 0.3,
},
{
name: "zero popularity track",
track: &models.Track{
PlayCount: 0,
LikeCount: 0,
},
wantMin: 0.0,
wantMax: 0.1,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
score := service.calculatePopularityScore(tt.track)
if score < tt.wantMin || score > tt.wantMax {
t.Errorf("calculatePopularityScore() = %v, want between %v and %v", score, tt.wantMin, tt.wantMax)
}
if score < 0 || score > 1 {
t.Errorf("calculatePopularityScore() = %v, want score between 0 and 1", score)
}
})
}
}
func TestTrackRecommendationService_calculateRecencyScore(t *testing.T) {
db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
if err != nil {
t.Fatalf("Failed to open database: %v", err)
}
service := NewTrackRecommendationService(db, zap.NewNop())
now := time.Now()
tests := []struct {
name string
track *models.Track
wantMin float64
wantMax float64
}{
{
name: "very recent track",
track: &models.Track{
CreatedAt: now.Add(-1 * time.Hour),
},
wantMin: 0.9,
wantMax: 1.0,
},
{
name: "recent track (15 days old)",
track: &models.Track{
CreatedAt: now.Add(-15 * 24 * time.Hour),
},
wantMin: 0.4,
wantMax: 0.6,
},
{
name: "old track (30+ days)",
track: &models.Track{
CreatedAt: now.Add(-35 * 24 * time.Hour),
},
wantMin: 0.0,
wantMax: 0.2,
},
{
name: "zero time track",
track: &models.Track{
CreatedAt: time.Time{},
},
wantMin: 0.0,
wantMax: 0.0,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
score := service.calculateRecencyScore(tt.track)
if score < tt.wantMin || score > tt.wantMax {
t.Errorf("calculateRecencyScore() = %v, want between %v and %v", score, tt.wantMin, tt.wantMax)
}
if score < 0 || score > 1 {
t.Errorf("calculateRecencyScore() = %v, want score between 0 and 1", score)
}
})
}
}
func TestGetKeys(t *testing.T) {
tests := []struct {
name string
m map[uuid.UUID]bool
want int
}{
{
name: "empty map",
m: map[uuid.UUID]bool{},
want: 0,
},
{
name: "map with one key",
m: map[uuid.UUID]bool{
uuid.New(): true,
},
want: 1,
},
{
name: "map with multiple keys",
m: map[uuid.UUID]bool{
uuid.New(): true,
uuid.New(): true,
uuid.New(): true,
},
want: 3,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got := getKeys(tt.m)
if len(got) != tt.want {
t.Errorf("getKeys() returned %d keys, want %d", len(got), tt.want)
}
})
}
}
func TestTrackRecommendationParams_Defaults(t *testing.T) {
params := TrackRecommendationParams{}
// Test that defaults are applied in GetRecommendations method
if params.Limit <= 0 {
params.Limit = 20
}
if params.Limit > 100 {
params.Limit = 100
}
if params.MinScore <= 0 {
params.MinScore = 0.1
}
if params.Limit != 20 {
t.Errorf("Expected default limit 20, got %d", params.Limit)
}
if params.MinScore != 0.1 {
t.Errorf("Expected default min score 0.1, got %f", params.MinScore)
}
}
// Note: Full integration tests would require:
// 1. A real database with test data (tracks, users, likes, plays)
// 2. Verification of collaborative filtering algorithm
// 3. Verification of content-based filtering
// 4. Performance testing with large datasets
//
// Example integration test structure:
// func TestTrackRecommendationService_GetRecommendations_Integration(t *testing.T) {
// // Setup test database
// db := setupTestDB(t)
// defer cleanupTestDB(t, db)
//
// // Create test data
// userID := createTestUser(t, db)
// tracks := createTestTracks(t, db, 50)
// createTestLikes(t, db, userID, tracks[:10])
// createTestPlays(t, db, userID, tracks[10:20])
//
// service := NewTrackRecommendationService(db, zap.NewNop())
//
// ctx := context.Background()
// params := TrackRecommendationParams{
// UserID: userID,
// Limit: 10,
// }
//
// recommendations, err := service.GetRecommendations(ctx, params)
// if err != nil {
// t.Fatalf("GetRecommendations() error = %v", err)
// }
//
// if len(recommendations) == 0 {
// t.Error("GetRecommendations() returned no recommendations")
// }
//
// // Verify recommendations are sorted by score
// for i := 1; i < len(recommendations); i++ {
// if recommendations[i].Score > recommendations[i-1].Score {
// t.Error("Recommendations are not sorted by score")
// }
// }
// }