我正在为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.gos3.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.gos3 package调用了方法。
我想知道如何模拟s3软件包中的DeleteS3Image,以避免错误和成功测试。

这是在dao_test.go上运行go test -v时的错误。

$ 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
分析解答

在您的设置中,您确实会调用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,以帮助您生成模拟。