DeviceRepository单测实现

This commit is contained in:
2025-09-12 16:36:44 +08:00
parent d09fb815bb
commit 8acefb2dd3
3 changed files with 151 additions and 21 deletions

View File

@@ -0,0 +1,107 @@
package repository_test
import (
"encoding/json"
"strconv"
"testing"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/models"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/repository"
"github.com/stretchr/testify/assert"
"gorm.io/gorm"
)
func TestGormDeviceRepository(t *testing.T) {
db := setupTestDB(t)
repo := repository.NewGormDeviceRepository(db)
// --- 准备测试数据 ---
loraProps, _ := json.Marshal(models.LoraProperties{LoraAddress: "0xABCD"})
busProps, _ := json.Marshal(models.BusProperties{BusID: 1, BusAddress: 10})
areaController := &models.Device{
Name: "1号猪舍主控",
Type: models.DeviceTypeAreaController,
Location: "1号猪舍",
Properties: loraProps,
}
t.Run("创建 - 成功创建区域主控", func(t *testing.T) {
err := repo.Create(areaController)
assert.NoError(t, err)
assert.NotZero(t, areaController.ID, "创建后应获得一个非零ID")
assert.Nil(t, areaController.ParentID, "区域主控的 ParentID 应为 nil")
})
var createdDevice *models.Device
t.Run("通过ID查找 - 成功找到已创建的设备", func(t *testing.T) {
var err error
createdDevice, err = repo.FindByID(areaController.ID)
assert.NoError(t, err)
assert.NotNil(t, createdDevice)
assert.Equal(t, areaController.Name, createdDevice.Name)
})
t.Run("通过字符串ID查找 - 使用有效字符串ID找到设备", func(t *testing.T) {
foundDevice, err := repo.FindByIDString(strconv.FormatUint(uint64(areaController.ID), 10))
assert.NoError(t, err)
assert.NotNil(t, foundDevice)
assert.Equal(t, areaController.ID, foundDevice.ID)
})
t.Run("通过字符串ID查找 - 使用无效字符串ID", func(t *testing.T) {
_, err := repo.FindByIDString("invalid-id")
assert.Error(t, err, "使用无效ID字符串应返回错误")
})
// 创建一个子设备
childDevice := &models.Device{
Name: "1号猪舍温度传感器",
Type: models.DeviceTypeDevice,
SubType: models.SubTypeSensorTemp,
ParentID: &areaController.ID,
Location: "1号猪舍东侧",
Properties: busProps,
}
t.Run("创建 - 成功创建子设备", func(t *testing.T) {
err := repo.Create(childDevice)
assert.NoError(t, err)
assert.NotZero(t, childDevice.ID)
assert.NotNil(t, childDevice.ParentID)
assert.Equal(t, areaController.ID, *childDevice.ParentID)
})
t.Run("通过父ID列出 - 找到子设备", func(t *testing.T) {
children, err := repo.ListByParentID(&areaController.ID)
assert.NoError(t, err)
assert.Len(t, children, 1, "应找到一个子设备")
assert.Equal(t, childDevice.ID, children[0].ID)
})
t.Run("通过父ID列出 - 找到顶层设备", func(t *testing.T) {
parents, err := repo.ListByParentID(nil)
assert.NoError(t, err)
assert.Len(t, parents, 1, "应找到一个顶层设备")
assert.Equal(t, areaController.ID, parents[0].ID)
})
t.Run("更新 - 成功更新设备信息", func(t *testing.T) {
childDevice.Location = "1号猪舍西侧"
err := repo.Update(childDevice)
assert.NoError(t, err)
updatedDevice, _ := repo.FindByID(childDevice.ID)
assert.Equal(t, "1号猪舍西侧", updatedDevice.Location)
})
t.Run("删除 - 成功删除设备", func(t *testing.T) {
err := repo.Delete(childDevice.ID)
assert.NoError(t, err)
// 验证设备已被软删除
_, err = repo.FindByID(childDevice.ID)
assert.Error(t, err, "删除后应无法找到设备")
assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 RecordNotFound")
})
}

View File

@@ -0,0 +1,38 @@
package repository_test
import (
"os"
"testing"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/models"
"github.com/stretchr/testify/assert"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
)
// setupTestDB 是一个共享的辅助函数,用于为集成测试创建一个干净的、内存中的 SQLite 数据库实例。
func setupTestDB(t *testing.T) *gorm.DB {
// "file::memory:?cache=shared" 是 GORM 连接内存 SQLite 的标准方式,确保在同一测试中的不同连接可以访问相同的数据。
db, err := gorm.Open(sqlite.Open("file::memory:?cache=shared"), &gorm.Config{})
assert.NoError(t, err, "连接内存数据库不应出错")
// 自动迁移所有需要的表结构
err = db.AutoMigrate(&models.User{}, &models.Device{})
assert.NoError(t, err, "数据库迁移不应出错")
return db
}
// TestMain 是一个特殊的函数,它会在包内的所有测试运行之前被调用。
// 我们可以在这里进行一些全局的设置和清理工作。
func TestMain(m *testing.M) {
// 在所有测试运行前可以执行一些设置代码
// 运行包中的所有测试
code := m.Run()
// 在所有测试运行后可以执行一些清理代码
// 退出测试
os.Exit(code)
}

View File

@@ -7,24 +7,9 @@ import (
"git.huangwc.com/pig/pig-farm-controller/internal/infra/models" "git.huangwc.com/pig/pig-farm-controller/internal/infra/models"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/repository" "git.huangwc.com/pig/pig-farm-controller/internal/infra/repository"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"gorm.io/driver/sqlite"
"gorm.io/gorm" "gorm.io/gorm"
) )
// setupTestDB 是一个辅助函数,用于为每个测试创建一个
// 干净的、内存中的 SQLite 数据库实例。
func setupTestDB(t *testing.T) *gorm.DB {
// "file::memory:?cache=shared" 是 GORM 连接内存 SQLite 的标准方式
db, err := gorm.Open(sqlite.Open("file::memory:?cache=shared"), &gorm.Config{})
assert.NoError(t, err, "连接内存数据库不应出错")
// 自动迁移 User 表结构
err = db.AutoMigrate(&models.User{})
assert.NoError(t, err, "数据库迁移不应出错")
return db
}
func TestGormUserRepository(t *testing.T) { func TestGormUserRepository(t *testing.T) {
db := setupTestDB(t) db := setupTestDB(t)
repo := repository.NewGormUserRepository(db) repo := repository.NewGormUserRepository(db)
@@ -35,7 +20,7 @@ func TestGormUserRepository(t *testing.T) {
Password: plainPassword, // 我们提供的是明文密码 Password: plainPassword, // 我们提供的是明文密码
} }
t.Run("Create - 成功创建并验证密码哈希", func(t *testing.T) { t.Run("创建 - 成功创建并验证密码哈希", func(t *testing.T) {
err := repo.Create(userToCreate) err := repo.Create(userToCreate)
assert.NoError(t, err) assert.NoError(t, err)
@@ -53,7 +38,7 @@ func TestGormUserRepository(t *testing.T) {
assert.True(t, savedUser.CheckPassword(plainPassword), "存储的密码哈希应该能与原明文匹配") assert.True(t, savedUser.CheckPassword(plainPassword), "存储的密码哈希应该能与原明文匹配")
}) })
t.Run("Create - 用户名冲突", func(t *testing.T) { t.Run("创建 - 用户名冲突", func(t *testing.T) {
// 尝试创建一个同名用户 // 尝试创建一个同名用户
duplicateUser := &models.User{Username: "testuser", Password: "anypassword"} duplicateUser := &models.User{Username: "testuser", Password: "anypassword"}
err := repo.Create(duplicateUser) err := repo.Create(duplicateUser)
@@ -64,7 +49,7 @@ func TestGormUserRepository(t *testing.T) {
assert.Contains(t, err.Error(), "UNIQUE constraint failed: users.username", "错误信息应包含唯一键冲突") assert.Contains(t, err.Error(), "UNIQUE constraint failed: users.username", "错误信息应包含唯一键冲突")
}) })
t.Run("FindByUsername - 找到用户", func(t *testing.T) { t.Run("按用户名查找 - 找到用户", func(t *testing.T) {
foundUser, err := repo.FindByUsername("testuser") foundUser, err := repo.FindByUsername("testuser")
assert.NoError(t, err) assert.NoError(t, err)
assert.NotNil(t, foundUser) assert.NotNil(t, foundUser)
@@ -72,20 +57,20 @@ func TestGormUserRepository(t *testing.T) {
assert.Equal(t, "testuser", foundUser.Username) assert.Equal(t, "testuser", foundUser.Username)
}) })
t.Run("FindByUsername - 未找到用户", func(t *testing.T) { t.Run("按用户名查找 - 未找到用户", func(t *testing.T) {
_, err := repo.FindByUsername("nonexistent") _, err := repo.FindByUsername("nonexistent")
assert.Error(t, err, "查找不存在的用户应该返回错误") assert.Error(t, err, "查找不存在的用户应该返回错误")
assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 gorm.ErrRecordNotFound") assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 gorm.ErrRecordNotFound")
}) })
t.Run("FindByID - 找到用户", func(t *testing.T) { t.Run("按ID查找 - 找到用户", func(t *testing.T) {
foundUser, err := repo.FindByID(userToCreate.ID) foundUser, err := repo.FindByID(userToCreate.ID)
assert.NoError(t, err) assert.NoError(t, err)
assert.NotNil(t, foundUser) assert.NotNil(t, foundUser)
assert.Equal(t, userToCreate.ID, foundUser.ID) assert.Equal(t, userToCreate.ID, foundUser.ID)
}) })
t.Run("FindByID - 未找到用户", func(t *testing.T) { t.Run("按ID查找 - 未找到用户", func(t *testing.T) {
_, err := repo.FindByID(99999) _, err := repo.FindByID(99999)
assert.Error(t, err, "查找不存在的ID应该返回错误") assert.Error(t, err, "查找不存在的ID应该返回错误")
assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 gorm.ErrRecordNotFound") assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 gorm.ErrRecordNotFound")