当前位置:首页 > 文章列表 > Golang > Go问答 > 接口“未定义”:模拟测试文件中是否已定义?

接口“未定义”:模拟测试文件中是否已定义?

来源:stackoverflow 2024-03-02 16:27:26 0浏览 收藏

本篇文章给大家分享《接口“未定义”:模拟测试文件中是否已定义?》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。

问题内容

这是一个错误,我试图用一个最小的例子来重现,但到目前为止还没有成功。 go 模块类似于以下内容:

.
├── go.mod
└── handler
    ├── handler.go
    ├── handler_test.go
    └── mock_handler.go

其中 handler.go 为空(仅包含 package handler),handler_test.go 包含 handler 接口定义(与 go 的 http.handler 相同)和占位符测试,

package handler

import (
    "net/http"
    "testing"
)

type handler interface {
    servehttp(http.responsewriter, *http.request)
}

func testmockhandler(t *testing.t) {
    mockhandler := mockhandler{}
    t.log(mockhandler)
}

并且mock_handler.go包含一个mockhandler结构,它实现了handler接口并使用moq生成:

// code generated by moq; do not edit.
// github.com/matryer/moq

package handler

import (
    "net/http"
    "sync"
)

var (
    lockmockhandlerservehttp sync.rwmutex
)

// ensure, that mockhandler does implement handler.
// if this is not the case, regenerate this file with moq.
var _ handler = &mockhandler{}

// mockhandler is a mock implementation of handler.
//
//     func testsomethingthatuseshandler(t *testing.t) {
//
//         // make and configure a mocked handler
//         mockedhandler := &mockhandler{
//             servehttpfunc: func(in1 http.responsewriter, in2 *http.request)  {
//                 panic("mock out the servehttp method")
//             },
//         }
//
//         // use mockedhandler in code that requires handler
//         // and then make assertions.
//
//     }
type mockhandler struct {
    // servehttpfunc mocks the servehttp method.
    servehttpfunc func(in1 http.responsewriter, in2 *http.request)

    // calls tracks calls to the methods.
    calls struct {
        // servehttp holds details about calls to the servehttp method.
        servehttp []struct {
            // in1 is the in1 argument value.
            in1 http.responsewriter
            // in2 is the in2 argument value.
            in2 *http.request
        }
    }
}

// servehttp calls servehttpfunc.
func (mock *mockhandler) servehttp(in1 http.responsewriter, in2 *http.request) {
    if mock.servehttpfunc == nil {
        panic("mockhandler.servehttpfunc: method is nil but handler.servehttp was just called")
    }
    callinfo := struct {
        in1 http.responsewriter
        in2 *http.request
    }{
        in1: in1,
        in2: in2,
    }
    lockmockhandlerservehttp.lock()
    mock.calls.servehttp = append(mock.calls.servehttp, callinfo)
    lockmockhandlerservehttp.unlock()
    mock.servehttpfunc(in1, in2)
}

// servehttpcalls gets all the calls that were made to servehttp.
// check the length with:
//     len(mockedhandler.servehttpcalls())
func (mock *mockhandler) servehttpcalls() []struct {
    in1 http.responsewriter
    in2 *http.request
} {
    var calls []struct {
        in1 http.responsewriter
        in2 *http.request
    }
    lockmockhandlerservehttp.rlock()
    calls = mock.calls.servehttp
    lockmockhandlerservehttp.runlock()
    return calls
}

为了生成mock_handler.go,我最初在handler.go中定义了handler,然后在handler目录中运行命令

moq -out mock_handler.go . handler

我随后将 handler 接口定义移至 handler_test.go,因为它仅用于测试。

在这个简化的示例中,我可以在根目录中以包列表模式运行 go test

~/g/s/g/k/mockhandler> go test ./... -count=1
ok      github.com/kurtpeek/mockhandler/handler 0.448s

我的“实际”模块具有类似的结构,类似于以下内容:

.
├── cmd
│   └── root.go
├── images
├── main.go
└── vpp
    ├── ensure_license_test.go
    └── mock_handler.go

handler 接口在 ensure_license_test.go 中的定义方式与简化模块中 handler_test.go 中的定义方式完全相同; ensure_license_test.go 的开头如下:

package vpp

import (
    "encoding/json"
    "io/ioutil"
    "net/http"
    "net/http/httptest"
    "net/url"
    "testing"

    "github.com/stretchr/testify/assert"
    "github.com/stretchr/testify/require"
)

type handler interface {
    servehttp(http.responsewriter, *http.request)
}

type mocktestserver struct {
    testserver  *httptest.server
    mockhandler *mockhandler
}

mock_handler.go 也与简化模块中的 mock_handler.go 完全相同(除了包名)。

但是,当我在“实际”模块的根目录中运行 go test ./... 时,我收到 undefined 错误:handler

~/g/s/g/f/vpp-client> go test ./... -count=1
# github.com/fleetsmith/vpp-client/vpp
vpp/mock_handler.go:17:7: undefined: handler
ok      github.com/fleetsmith/vpp-client/vpp    0.128s

奇怪的是,当我从 vpp 包中运行它时,它通过了:

> go test ./... -count=1
ok      github.com/fleetsmith/vpp-client/vpp    0.601s

像第一个示例一样,当从根目录以包列表模式运行 go test 时,无法定义 handler 的定义,这可能是什么原因?


解决方案


事实证明,测试失败的是 cmd 包,因为无法从 handler 包中的测试文件导入 handler 接口。因此,我更改了 mock_handler.go 的第 17 行以使用 http.handler 而不是 handler

var _ http.handler = &mockhandler{}

现在测试通过了:

~/g/s/g/f/vpp-client> go test ./...
?       github.com/fleetsmith/vpp-client    [no test files]
?       github.com/fleetsmith/vpp-client/cmd    [no test files]
ok      github.com/fleetsmith/vpp-client/vpp    0.462s

我还可以从 ensure_license_test.go 中删除 handler 接口的定义,因为我现在直接使用 http 标准库中的定义。

这种方法的缺点是它需要编辑由 moq 自动生成的代码,但是,我无法弄清楚如何运行 moq 来模拟 go 标准库中的接口,并且无论如何这个接口是不太可能改变。

理论要掌握,实操不能落!以上关于《接口“未定义”:模拟测试文件中是否已定义?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
使用 Golang 对 PGP 进行解密操作使用 Golang 对 PGP 进行解密操作
上一篇
使用 Golang 对 PGP 进行解密操作
解答十大Discuz导航栏修改常见问题
下一篇
解答十大Discuz导航栏修改常见问题
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 毕业宝AIGC检测:AI生成内容检测工具,助力学术诚信
    毕业宝AIGC检测
    毕业宝AIGC检测是“毕业宝”平台的AI生成内容检测工具,专为学术场景设计,帮助用户初步判断文本的原创性和AI参与度。通过与知网、维普数据库联动,提供全面检测结果,适用于学生、研究者、教育工作者及内容创作者。
    23次使用
  • AI Make Song:零门槛AI音乐创作平台,助你轻松制作个性化音乐
    AI Make Song
    AI Make Song是一款革命性的AI音乐生成平台,提供文本和歌词转音乐的双模式输入,支持多语言及商业友好版权体系。无论你是音乐爱好者、内容创作者还是广告从业者,都能在这里实现“用文字创造音乐”的梦想。平台已生成超百万首原创音乐,覆盖全球20个国家,用户满意度高达95%。
    33次使用
  • SongGenerator.io:零门槛AI音乐生成器,快速创作高质量音乐
    SongGenerator
    探索SongGenerator.io,零门槛、全免费的AI音乐生成器。无需注册,通过简单文本输入即可生成多风格音乐,适用于内容创作者、音乐爱好者和教育工作者。日均生成量超10万次,全球50国家用户信赖。
    30次使用
  •  BeArt AI换脸:免费在线工具,轻松实现照片、视频、GIF换脸
    BeArt AI换脸
    探索BeArt AI换脸工具,免费在线使用,无需下载软件,即可对照片、视频和GIF进行高质量换脸。体验快速、流畅、无水印的换脸效果,适用于娱乐创作、影视制作、广告营销等多种场景。
    33次使用
  • SEO标题协启动:AI驱动的智能对话与内容生成平台 - 提升创作效率
    协启动
    SEO摘要协启动(XieQiDong Chatbot)是由深圳协启动传媒有限公司运营的AI智能服务平台,提供多模型支持的对话服务、文档处理和图像生成工具,旨在提升用户内容创作与信息处理效率。平台支持订阅制付费,适合个人及企业用户,满足日常聊天、文案生成、学习辅助等需求。
    36次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码