doutuan6158 2017-11-14 14:14
浏览 43
已采纳

如何使用结构/接口模拟依赖关系进行测试

I'm new to go here... my objective is to unit test that a status is being updated in my ready(). I've been looking at https://engineering.aircto.com/writing-testable-code-in-golang/ and trying to figure out how to adapt what they're doing to my use case, filling in gaps of golang knowledge where I can.

I'm getting the error cannot use fakeSession (type *FakeSession) as type *discordgo.Session in argument to ready but I'm not sure why I'm getting this error.

main.go

import (
    "fmt"
    "os"
    "os/signal"
    "syscall"

    "github.com/bwmarrin/discordgo"
)

var (
    // bot token used for this bot when connecting
    token    = os.Getenv("DISCORD_BOT_TOKEN")
    status   = os.Getenv("BOT_STATUS")
)

func main() {
    // initiate Discord bot

    // Register ready as a callback for the ready events.
    discordConnection.AddHandler(ready)

    // running the app, waiting to receive a close signal
}

// This function will be called (due to AddHandler above) when the bot receives
// the "ready" event from Discord.
func ready(session *discordgo.Session, event *discordgo.Ready) {

    // Set the playing status.
    session.UpdateStatus(0, status)
}

main_test.go

type FakeSession struct {
    status  string
    idle    int
}

func (f *FakeSession) UpdateStatus(idle int, game string) error {
    f.idle, f.status = idle, game
    return nil
}

func TestStatusIsUpdated(t *testing.T) {
    readyDependency := &discordgo.Ready{}
    fakeSession := &FakeSession{}

    ready(fakeSession, readyDependency)

    // @todo assert that idle/game status were set to correct values
}
  • 写回答

1条回答 默认 最新

  • dongxian7489 2017-11-14 14:35
    关注

    As @Andrew pointed out discordgo.Session is a go struct (from the docs link you posted type Session struct {)

    structs are Concrete types in go and are unable to be substituted. The only argument go compiler will allow for ready is a pointer to a session.

    To break this dependency you can create a custom interface owned and controlled by your project using the methods that you need. This will allow you to create and call ready with a fake structure for your tests.

    Sometimes 3rd party libraries already have interfaces so it's usually worth-while to scan their godoc to see which interfaces are available before creating your own.


    But if you have to create your own for testing (and I find myself regularly having to do this), it might look like:

    type StatusUpdater interface {
       UpdateStatus(int, string)
    }
    
    // This function will be called (due to AddHandler above) when the bot receives
    // the "ready" event from Discord.
    func ready(s StatusUpdater, event *discordgo.Ready) {
    
        // Set the playing status.
        s.UpdateStatus(0, status)
    }
    

    Now the dependency on discordgo.Session has been broken, and your test code can call ready function with its fake session, and then make assertions on it!

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥100 Jenkins自动化部署—悬赏100元
  • ¥15 关于#python#的问题:求帮写python代码
  • ¥20 MATLAB画图图形出现上下震荡的线条
  • ¥15 关于#windows#的问题:怎么用WIN 11系统的电脑 克隆WIN NT3.51-4.0系统的硬盘
  • ¥15 perl MISA分析p3_in脚本出错
  • ¥15 k8s部署jupyterlab,jupyterlab保存不了文件
  • ¥15 ubuntu虚拟机打包apk错误
  • ¥199 rust编程架构设计的方案 有偿
  • ¥15 回答4f系统的像差计算
  • ¥15 java如何提取出pdf里的文字?