草庐IT

go - 用模拟编写单元测试

coder 2024-07-08 原文

我在里面写了一个REST api。现在我想通过隔离不同的组件来对其进行单元测试。

项目结构如下所示,有两个主要包

├── main.go
├── routes
│   ├── routes.go
│   └── routes_test.go
├── db
│   └── db.go

ma​​in.go:项目的主要入口点

routes/routes.go:HTTP路由处理包

db/db.go:数据库处理器包

现在,当我测试 http 路由时,我只想测试请求是否被路由到正确的处理程序函数,然后它们会做出相应的响应。在实际应用程序中,处理函数实际上会插入/更新数据库,但我不想在测试时这样做。因此,如果我可以模拟一个数据库对象并使我的处理程序函数针对该对象进行连接/读取/写入,但我不确定该怎么做。

在测试数据库处理程序包 db.go 时,我认为我不能模拟它,我必须设置一个测试数据库,我应该对其进行查询。

这是我的文件的样子。

路线/routes.go

    package routes

    import (
        "cfengine-service/db"
        "encoding/json"
        "net/http"
        "strings"

        log "github.com/sirupsen/logrus"

        "github.com/gorilla/mux"
    )

    // Route - struct modal for routes
    type Route struct {
        Name        string
        EndPoint    string
        Methods     []string
        HandlerFunc http.HandlerFunc
    }
    var dbHandler = &db.Mongo{}

    // SetupRouter - returns the mux router
    func SetupRouter() *mux.Router {
        err := dbHandler.Connect()
        if err != nil {
            log.Fatal(err)
        }
        router := mux.NewRouter()
        for _, route := range SetUpRoutes() {
            router.HandleFunc(route.EndPoint, route.HandlerFunc).Methods(route.Methods...)
        }
        return router

    }
    func SetUpRoutes() []Route {
        return []Route{
            {
                "Host config Management",
                "/configs/group",
                []string{"GET", "POST"},
                GroupConfig,
            },
    }

func GroupConfig(w http.ResponseWriter, r *http.Request) {
    reqMethod := r.Method
    if reqMethod == "GET" {
        GetGroupConfig(w, r)
    } else if reqMethod == "POST" {
        NewGroupConfig(w, r)
    }

}

func GetGroupConfig(w http.ResponseWriter, r *http.Request) {
    w.Header().Set("Content-Type", "application/json")
    // g := r.URL.Query().Get("groupname")
    g := r.FormValue("groupname")
    if g == "" {
        if data, err := dbHandler.ReadAllGroups(); StatusWriter(w, err) {
            _ = json.NewEncoder(w).Encode(data)
        }
    } else {
        if data, err := dbHandler.ReadOneGroup(g); StatusWriter(w, err) {
            _ = json.NewEncoder(w).Encode(data)
        }
    }

}

// NewGroupConfig - end point for creating new config group
func NewGroupConfig(w http.ResponseWriter, r *http.Request) {
    w.Header().Set("content-type", "application/json")
    var group db.GroupConfig
    err := json.NewDecoder(r.Body).Decode(&group)
    StatusWriter(w, err)

    err = dbHandler.CreateGroupConfig(&group)
    StatusWriter(w, err)
}

db/db.go

    package db

    import (
        "cfengine-service/config"
        "cfengine-service/helpers"
        "context"
        "fmt"
        "time"

        log "github.com/sirupsen/logrus"
        "go.mongodb.org/mongo-driver/bson"
        "go.mongodb.org/mongo-driver/mongo"
        "go.mongodb.org/mongo-driver/mongo/options"
    )

    type HostConfig struct {
        Hostname  string            `bson:"_id" json:"hostname"`
        Type      string            `bson:"type" json:"type"`
        Groups    []string          `bson:"groups" json:"groups"`
        Overrides map[string]string `bson:"overrides" json:"overrides"`
        Excludes  []string          `bson:"excludes" json:"excludes"`
        Data      map[string]string `bson:"data" json:"data"`
    }


    // Mongo collection
    type Mongo struct {
        collection *mongo.Collection
    }

    // Storage is interface for db operations
    type Storage interface {
        Connect() error
        ReadAll() ([]HostConfig, error)
        ReadOne() (*HostConfig, error)
        CreateHostConfig(h *HostConfig) error
        CreateGroupConfig(g *GroupConfig) error
        DeleteHostConfig(h *HostConfig) error
        UpdateHostGrops(h string, g string, opr string) error
        validateGroup(ctx context.Context, val string) error
    }

    func (m *Mongo) Connect() error {
        ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
        defer cancel()
        //client, err := mongo.NewClient(config.Config.MongoURI)
        client, err := mongo.Connect(ctx, options.Client().ApplyURI(config.Config.MongoURI))
        if err != nil {
            return err
        }

        log.Info("Connected to MongoDB.....")
        m.collection = client.Database(config.Config.MongoDbName).Collection(config.Config.MongoCollectionName)
        return nil
    }
func (m *Mongo) CreateGroupConfig(g *GroupConfig) error {
    ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
    defer cancel()
    data, err := bson.Marshal(g)
    if err != nil {
        log.Error(err.Error())
        return err
    }
    _, err = m.collection.InsertOne(ctx, data)
    if err != nil {
        return err
    }
    return nil
}

func (m *Mongo) ReadOneGroup(g string) (GroupConfig, error) {
    ctx, cancel := context.WithTimeout(context.Background(), time.Second*10)
    defer cancel()
    var group GroupConfig
    err := m.collection.FindOne(ctx, bson.M{"_id": g}).Decode(&group)
    if err != nil {
        return GroupConfig{}, err
    }

    return group, nil
}

func (m *Mongo) ReadAllGroups() ([]GroupConfig, error) {
    var configs []GroupConfig
    ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
    defer cancel()
    cur, err := m.collection.Find(ctx, bson.M{"type": "group"})
    if err != nil {
        return nil, err
    }
    defer cur.Close(ctx)
    for cur.Next(ctx) {
        var record GroupConfig
        if err = cur.Decode(&record); err != nil {
            return nil, err
        }
        configs = append(configs, record)
    }
    return configs, nil
}

因此,使用 mockgen 包,我在 db.go 中为 Storage 接口(interface)生成了模拟,但我不确定如何开始使用那个。这是 mock_db.go 的样子。

mock_db.go

type MockStorage struct {
    ctrl     *gomock.Controller
    recorder *MockStorageMockRecorder
}

// MockStorageMockRecorder is the mock recorder for MockStorage
type MockStorageMockRecorder struct {
    mock *MockStorage
}

// NewMockStorage creates a new mock instance
func NewMockStorage(ctrl *gomock.Controller) *MockStorage {
    mock := &MockStorage{ctrl: ctrl}
    mock.recorder = &MockStorageMockRecorder{mock}
    return mock
}

// EXPECT returns an object that allows the caller to indicate expected use
func (m *MockStorage) EXPECT() *MockStorageMockRecorder {
    return m.recorder
}

// Connect mocks base method
func (m *MockStorage) Connect() error {
    m.ctrl.T.Helper()
    ret := m.ctrl.Call(m, "Connect")
    ret0, _ := ret[0].(error)
    return ret0
}

// Connect indicates an expected call of Connect
func (mr *MockStorageMockRecorder) Connect() *gomock.Call {
    mr.mock.ctrl.T.Helper()
    return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Connect", reflect.TypeOf((*MockStorage)(nil).Connect))
}
func (m *MockStorage) CreateGroupConfig(arg0 *db.GroupConfig) error {
    m.ctrl.T.Helper()
    ret := m.ctrl.Call(m, "CreateGroupConfig", arg0)
    ret0, _ := ret[0].(error)
    return ret0
}

// CreateGroupConfig indicates an expected call of CreateGroupConfig
func (mr *MockStorageMockRecorder) CreateGroupConfig(arg0 interface{}) *gomock.Call {
    mr.mock.ctrl.T.Helper()
    return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "CreateGroupConfig", reflect.TypeOf((*MockStorage)(nil).CreateGroupConfig), arg0)
}

所以任何人都可以在这里指导我或给我一些提示让我朝着正确的方向前进。

最佳答案

您在测试处理程序时犯的错误是您在 SetupRouter 函数中创建了所谓的“单例对象”,因为您的应用程序所需的所有依赖项都应在 中实例化main.go 并将它们作为您需要它们的接口(interface)。还要定义小型接口(interface) - 理想的接口(interface)大小是 1 或 2 个方法。

//Storage 是你已经有的接口(interface),这样你可以在单元测试时传递你的模拟对象,Router 也应该是一个接口(interface)。

func SetupRouter(db Storage, router Router) *mux.Router {

        err := db.Connect()
        if err != nil {
            log.Fatal(err)
        }
        //router := mux.NewRouter() - you do not need to do this now
        // then perform whateven actions you want to do with router
}

所以上面的代码片段只是一个示例,说明如何模拟您的案例数据库中的依赖项,但正如@Adrian 在评论中所说,如果此代码是生产代码,则需要进行大量重构。

关于go - 用模拟编写单元测试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56187049/

有关go - 用模拟编写单元测试的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  3. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  4. ruby - Ruby 的 Hash 在比较键时使用哪种相等性测试? - 2

    我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的:classAattr_reader:xdefinitialize(inner)@inner=innerenddefx;@inner.x;enddef==(other)@inner.x==other.xendenda=A.new(o)#oisjustanyobjectthatallowso.xb=A.new(o)h={a=>5}ph[a]#5ph[b]#nil,shouldbe5ph[o]#nil,shouldbe5我试过==、===、eq?并散列所有无济于事。

  5. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  6. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  7. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  8. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

  9. ruby - 即使失败也继续进行多主机测试 - 2

    我已经构建了一些serverspec代码来在多个主机上运行一组测试。问题是当任何测试失败时,测试会在当前主机停止。即使测试失败,我也希望它继续在所有主机上运行。Rakefile:namespace:specdotask:all=>hosts.map{|h|'spec:'+h.split('.')[0]}hosts.eachdo|host|begindesc"Runserverspecto#{host}"RSpec::Core::RakeTask.new(host)do|t|ENV['TARGET_HOST']=hostt.pattern="spec/cfengine3/*_spec.r

  10. ruby-on-rails - 如何使辅助方法在 Rails 集成测试中可用? - 2

    我在app/helpers/sessions_helper.rb中有一个帮助程序文件,其中包含一个方法my_preference,它返回当前登录用户的首选项。我想在集成测试中访问该方法。例如,这样我就可以在测试中使用getuser_path(my_preference)。在其他帖子中,我读到这可以通过在测试文件中包含requiresessions_helper来实现,但我仍然收到错误NameError:undefinedlocalvariableormethod'my_preference'.我做错了什么?require'test_helper'require'sessions_hel

随机推荐