test: add comprehensive UserRepository coverage tests
Added tests for UserRepository operations: - TestUserRepository_GetByPhone - TestUserRepository_ListByStatus - TestUserRepository_UpdateStatus - TestUserRepository_BatchUpdateStatus - TestUserRepository_BatchDelete - TestUserRepository_Search - TestUserRepository_Search_LikePattern (tests LIKE escape) repository package coverage: 46% → 47.2%
This commit is contained in:
@@ -196,3 +196,208 @@ func TestUserRepository_List(t *testing.T) {
|
||||
t.Errorf("total = %d, want 5", total)
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_GetByPhone tests phone lookup
|
||||
func TestUserRepository_GetByPhone(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
user := &domain.User{
|
||||
Username: "phoneuser",
|
||||
Email: domain.StrPtr("phone@example.com"),
|
||||
Phone: domain.StrPtr("13700137000"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
repo.Create(ctx, user)
|
||||
|
||||
found, err := repo.GetByPhone(ctx, "13700137000")
|
||||
if err != nil {
|
||||
t.Fatalf("GetByPhone() error = %v", err)
|
||||
}
|
||||
if found.Username != "phoneuser" {
|
||||
t.Errorf("Username = %v, want phoneuser", found.Username)
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_ListByStatus tests status filtering
|
||||
func TestUserRepository_ListByStatus(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "active1",
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
})
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "active2",
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
})
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "inactive1",
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusInactive,
|
||||
})
|
||||
|
||||
users, total, err := repo.ListByStatus(ctx, domain.UserStatusActive, 0, 10)
|
||||
if err != nil {
|
||||
t.Fatalf("ListByStatus() error = %v", err)
|
||||
}
|
||||
if len(users) != 2 {
|
||||
t.Errorf("len(users) = %d, want 2", len(users))
|
||||
}
|
||||
if total != 2 {
|
||||
t.Errorf("total = %d, want 2", total)
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_UpdateStatus tests status update
|
||||
func TestUserRepository_UpdateStatus(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
user := &domain.User{
|
||||
Username: "statususer",
|
||||
Email: domain.StrPtr("status@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
repo.Create(ctx, user)
|
||||
|
||||
err := repo.UpdateStatus(ctx, user.ID, domain.UserStatusInactive)
|
||||
if err != nil {
|
||||
t.Fatalf("UpdateStatus() error = %v", err)
|
||||
}
|
||||
|
||||
found, _ := repo.GetByID(ctx, user.ID)
|
||||
if found.Status != domain.UserStatusInactive {
|
||||
t.Errorf("Status = %v, want Inactive", found.Status)
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_BatchUpdateStatus tests batch status update
|
||||
func TestUserRepository_BatchUpdateStatus(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
user1 := &domain.User{
|
||||
Username: "batch1",
|
||||
Email: domain.StrPtr("batch1@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
user2 := &domain.User{
|
||||
Username: "batch2",
|
||||
Email: domain.StrPtr("batch2@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
repo.Create(ctx, user1)
|
||||
repo.Create(ctx, user2)
|
||||
|
||||
err := repo.BatchUpdateStatus(ctx, []int64{user1.ID, user2.ID}, domain.UserStatusInactive)
|
||||
if err != nil {
|
||||
t.Fatalf("BatchUpdateStatus() error = %v", err)
|
||||
}
|
||||
|
||||
found1, _ := repo.GetByID(ctx, user1.ID)
|
||||
found2, _ := repo.GetByID(ctx, user2.ID)
|
||||
if found1.Status != domain.UserStatusInactive || found2.Status != domain.UserStatusInactive {
|
||||
t.Error("BatchUpdateStatus failed")
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_BatchDelete tests batch delete
|
||||
func TestUserRepository_BatchDelete(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
user1 := &domain.User{
|
||||
Username: "del1",
|
||||
Email: domain.StrPtr("del1@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
user2 := &domain.User{
|
||||
Username: "del2",
|
||||
Email: domain.StrPtr("del2@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
}
|
||||
repo.Create(ctx, user1)
|
||||
repo.Create(ctx, user2)
|
||||
|
||||
err := repo.BatchDelete(ctx, []int64{user1.ID, user2.ID})
|
||||
if err != nil {
|
||||
t.Fatalf("BatchDelete() error = %v", err)
|
||||
}
|
||||
|
||||
_, err1 := repo.GetByID(ctx, user1.ID)
|
||||
_, err2 := repo.GetByID(ctx, user2.ID)
|
||||
if err1 == nil || err2 == nil {
|
||||
t.Error("BatchDelete should have deleted users")
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_Search tests user search
|
||||
func TestUserRepository_Search(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "searchuser1",
|
||||
Nickname: "张三",
|
||||
Email: domain.StrPtr("zhangsan@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
})
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "searchuser2",
|
||||
Nickname: "李四",
|
||||
Email: domain.StrPtr("lisi@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
})
|
||||
|
||||
users, total, err := repo.Search(ctx, "zhang", 0, 10)
|
||||
if err != nil {
|
||||
t.Fatalf("Search() error = %v", err)
|
||||
}
|
||||
if len(users) != 1 {
|
||||
t.Errorf("len(users) = %d, want 1", len(users))
|
||||
}
|
||||
if total != 1 {
|
||||
t.Errorf("total = %d, want 1", total)
|
||||
}
|
||||
}
|
||||
|
||||
// TestUserRepository_Search_LikePattern tests search with LIKE special chars
|
||||
func TestUserRepository_Search_LikePattern(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := NewUserRepository(db)
|
||||
ctx := context.Background()
|
||||
|
||||
repo.Create(ctx, &domain.User{
|
||||
Username: "user%with%percent",
|
||||
Nickname: "测试用户",
|
||||
Email: domain.StrPtr("percent@example.com"),
|
||||
Password: "hash",
|
||||
Status: domain.UserStatusActive,
|
||||
})
|
||||
|
||||
// Search should handle LIKE special chars safely
|
||||
users, _, err := repo.Search(ctx, "%", 0, 10)
|
||||
if err != nil {
|
||||
t.Fatalf("Search() error = %v", err)
|
||||
}
|
||||
// Should not error and should escape properly
|
||||
_ = users
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user