我正在为api(golang)设置单元测试。
似乎使用了嘲讽。但是我不明白如何编码才能成功。
article
├ client
├ api
│ ├ main.go
│ ├ contoroller
│ │ ├ contoroller.go
│ │ └ contoroller_test.go
│ ├ service
│ │ ├ service.go
│ │ └ service_test.go
│ ├ dao
│ │ ├ dao.go
│ │ └ dao_test.go
│ ├ s3
│ │ ├ s3.go
│ │ └ s3_test.go
│ ├ go.mod
│ ├ go.sum
│ └ Dockerfile
├ nginx
└ docker-compose.yml
现在我正在尝试设置dao_test.go
但这失败了,因为dao.go
从s3.dao
调用了方法。
dao_test.go
package dao
// import
type DaoSuite struct {
suite.Suite
db *sql.DB
mock sqlmock.Sqlmock
dao *Dao
s3 *s3.S3
}
func (s *DaoSuite) SetupTest() {
var err error
s.db, s.mock, err = sqlmock.New()
s.Require().NoError(err)
s.dao = NewDao(s.db, s.s3)
}
func (s *DaoSuite) TestDeleteArticleDao() {
// some method
// here test fails because DeleteArticleDao calls method from another package.
s.dao.DeleteArticleDao("1")
}
func (s *DaoSuite) TearDownTest() {
s.db.Close()
s.Assert().NoError(s.mock.ExpectationsWereMet())
}
dao.go
package dao
// import
type Dao struct {
database *sql.DB
s3 *s3.S3
}
func NewDao(database *sql.DB, s3 *s3.S3) *Dao {
objs := &Dao{database: database, s3: s3}
return objs
}
func (d *Dao) DeleteArticleDao(id string) {
//generate imageName
//here calls method in package s3
//here test fails
d.s3.DeleteS3Image(imageName)
}
s3.go
package s3
//import
type S3 struct {
APPID string
SECRET string
}
type DaoInterface interface {
DeleteS3Image(imageName util.ImageName) error
}
func NewS3(appid, secret string) *S3 {
objs := &S3{APPID: appid, SECRET: secret}
return objs
}
func (objs *S3) DeleteS3Image(imageName util.ImageName) error {
// method
}
完整的源代码在这里(fix-test-dao):
https://github.com/jpskgc/article/tree/fix-test-dao
我希望dao_test.go
的测试成功。
但实际上是失败的,因为dao.go
从s3 package
调用了方法。
我想知道如何在软件包s3中模拟DeleteS3Image
以避免错误和成功测试。
在go test -v
上运行dao_test.go
时出现错误。
$ go test -v
--- FAIL: TestDaoSuite (0.00s)
--- FAIL: TestDaoSuite/TestDeleteArticleDao (0.00s)
dao_test.go:221:
Error Trace: dao_test.go:221
suite.go:122
panic.go:522
panic.go:82
signal_unix.go:390
s3.go:66
dao.go:74
dao_test.go:156
Error: Received unexpected error:
there is a remaining expectation which was not matched: ExpectedBegin => expecting database transaction Begin
Test: TestDaoSuite/TestDeleteArticleDao
suite.go:61: test panicked: runtime error: invalid memory address or nil pointer dereference
答案 0 :(得分:0)
在您的设置中,您确实会调用s.dao = NewDao(s.db, s.s3)
,但是您从未将s.s3
初始化为任何内容,因此s.dao.s3
仍然是nil
,这就是d.s3.DeleteS3Image(imageName)
恐慌的原因。
在Go中,为了能够模拟方法,调用该方法的值必须是接口,而不是具体类型。换句话说,在Go中模拟具体方法是不可能的。
所以具有这样的类型:
type Dao struct {
database *sql.DB
s3 *s3.S3
}
您只是不能模拟s3
。
您所能做的就是将s3
字段的类型更改为一种接口类型,您已经准备好了(s3.DaoInterface
)。
type Dao struct {
database *sql.DB
s3 s3.DaoInterface
}
现在您可以模拟s3
字段。
剩下的就是您要实现模拟并确保在测试设置过程中将s3
字段设置为模拟实现的实例。
type MockS3 struct{}
func (MockS3) DeleteS3Image(imageName util.ImageName) error {
// do whatever
return nil
}
func (s *DaoSuite) SetupTest() {
var err error
s.db, s.mock, err = sqlmock.New()
s.Require().NoError(err)
s.dao = NewDao(s.db, s.s3)
s.dao.s3 = MockS3{} // <- don't forget about me
}
由您决定如何实现模拟,但是如果您不熟悉模拟,我建议您看看https://github.com/golang/mock来帮助您生成模拟。