DeviceRepository单测实现
This commit is contained in:
107
internal/infra/repository/device_repository_test.go
Normal file
107
internal/infra/repository/device_repository_test.go
Normal 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")
|
||||
})
|
||||
}
|
||||
38
internal/infra/repository/main_test.go
Normal file
38
internal/infra/repository/main_test.go
Normal 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)
|
||||
}
|
||||
@@ -7,24 +7,9 @@ import (
|
||||
"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/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, "连接内存数据库不应出错")
|
||||
|
||||
// 自动迁移 User 表结构
|
||||
err = db.AutoMigrate(&models.User{})
|
||||
assert.NoError(t, err, "数据库迁移不应出错")
|
||||
|
||||
return db
|
||||
}
|
||||
|
||||
func TestGormUserRepository(t *testing.T) {
|
||||
db := setupTestDB(t)
|
||||
repo := repository.NewGormUserRepository(db)
|
||||
@@ -35,7 +20,7 @@ func TestGormUserRepository(t *testing.T) {
|
||||
Password: plainPassword, // 我们提供的是明文密码
|
||||
}
|
||||
|
||||
t.Run("Create - 成功创建并验证密码哈希", func(t *testing.T) {
|
||||
t.Run("创建 - 成功创建并验证密码哈希", func(t *testing.T) {
|
||||
err := repo.Create(userToCreate)
|
||||
assert.NoError(t, err)
|
||||
|
||||
@@ -53,7 +38,7 @@ func TestGormUserRepository(t *testing.T) {
|
||||
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"}
|
||||
err := repo.Create(duplicateUser)
|
||||
@@ -64,7 +49,7 @@ func TestGormUserRepository(t *testing.T) {
|
||||
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")
|
||||
assert.NoError(t, err)
|
||||
assert.NotNil(t, foundUser)
|
||||
@@ -72,20 +57,20 @@ func TestGormUserRepository(t *testing.T) {
|
||||
assert.Equal(t, "testuser", foundUser.Username)
|
||||
})
|
||||
|
||||
t.Run("FindByUsername - 未找到用户", func(t *testing.T) {
|
||||
t.Run("按用户名查找 - 未找到用户", func(t *testing.T) {
|
||||
_, err := repo.FindByUsername("nonexistent")
|
||||
assert.Error(t, err, "查找不存在的用户应该返回错误")
|
||||
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)
|
||||
assert.NoError(t, err)
|
||||
assert.NotNil(t, foundUser)
|
||||
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)
|
||||
assert.Error(t, err, "查找不存在的ID应该返回错误")
|
||||
assert.ErrorIs(t, err, gorm.ErrRecordNotFound, "错误类型应为 gorm.ErrRecordNotFound")
|
||||
|
||||
Reference in New Issue
Block a user