Skip to content

Methodman is a stub-based Denpendency Injection tool in Go

License

Notifications You must be signed in to change notification settings

jason-xxl/methodman

Repository files navigation

methodman

methodman is a stub-based Denpendency Injection tool in Go.

Supported Features

  • Stub-based dependency injection for test. It can hijack any method stub to produce fake response for test scenarios.
  • Optimised to minimise code overhead. So that the api of pkgs remain clean and won't be polluted for test requirement.
  • Supports parallel unittest. Mocking in one goroutine is invisible / isolated from any other goroutine.
  • Supports stub-injection with a temporary func, which could be useful for simulating timeout, panic, mock with internal state (via closure), or any other kind of side-effects.
  • Certain helper support to make conversion from integration-test to unit-test easier. See the sections below.

How to use?

Assuming in my_pkg, you have a method MyFunc that depends on MethodA in another pkg dep_pkg, like this,

package my_pkg

...

var MyFunc = func() (resp1, resp2 string){
	resp1, resp2 = dep_pkg.MethodA()
	return 
}

Now I'd like to write some unittest for MyFunc with mocking its dependency dep_pkg.MethodA.

1. Make sure methodman is installed.
go get -u github.com/jason-xxl/methodman
2. Register the method to be mocked (dep_pkg.MethodA) with a valid name.
func TestMain(m *testing.M) {
	flag.Parse()
	mm.EnableMock(&dep_pkg.MethodA, "dep_pkg.MethodA")
	os.Exit(m.Run())
}
3. Mock it in your test.
func TestNormalUse(t *testing.T) {

	defer mm.Init(t).CleanUp()
	
	mm.Expect(&dep_pkg.MethodA, "some fake response as 1st returned var", "some more, as 2nd retuened var")

	// Then you can receive above 2 value in your code path.
	// Inside MyFunc, "dep_pkg.MethodA" would be called, but since it's mocked, you will receive the fake response,
	// instead of executing the real logic of dep_pkg.MethodA.
	ret1, ret2 := MyFunc()
	
	if ret1 == "some fake response as 1st returned var" && ret2 == "some more, as 2nd retuened var" {
		t.Log("awesome! I received the fake responses!")
	}
	
	// 1. If all fake responses are consumed, the agent will fall back to original method.
	// 2. It doesn't matter next call of dep_pkg.MethodA is at which level, above fake value would be 
	//    received if it's in same goroutine.
}

How methodman works?

The stub based approach is simple.

  1. When registering the dependency method var, methodman will replace (monkey patch) the var with a manager object. It wraps the original method with a queue layer in front (one queue for one method in one goroutine).

  2. When you push a fake response to a method, the response will enter the queue of current goroutine.

  3. When the method endpoint is called (actually the manager object is called), it will check the queue of current goroutine. If the queue is non-empty, the method will response the fake response by consuming the queue. When queue is empty, the original method is called to provide a real response.

How to enable monkey patching (stub injection) in Golang?

In many dynamic languages it's easy to monkey patching object or methods. However, in Go, exported pkg method is not modifiable. So by default there's no formal way to monkey-patch. To get monkey patching works, pkg method have to be defined as an exported method variable to allow modify (in case you can control the code),

  var TheMethodToBeMocked = func(...){...}

Or, use a reference var in caller side like this (in case you can't control the code),

  var TheMethodToBeMocked = targetpkg.TargetMethod

Then you can monkey patch TheMethodToBeMocked for mocking.

Given the target method var is modifiable, methodman would replace TheMethodToBeMocked with a wrapper, who provides mockability that just overlays the original method.

How to convert integration-test into unittest? (For refactoring scenarios)

Assuming you have a test case that accesses external dependencies and already works fine, assuming no forking gorouting inside the logic, and now, you want to convert it into a unittest.

So the idea here is you run your integration-test once, capture those real outputs of depending methods, and use those output as sample to mock. Here's step by step.

1. Enable the CapturingLogger and register the methods that you want to mock.
func TestMain(m *testing.M) {
	flag.Parse()

	mm.SetLogger(mm.CapturingLogger)

	mm.EnableMock(&dep_pkg.MethodA, "dep_pkg1.MethodA")
	mm.EnableMock(&dep_pkg.MethodB, "dep_pkg2.MethodA")

	os.Exit(m.Run())
}
2. Run your test in verbose mode, you can get the real response in the form of usable code that you can insert into your code. This save human effort to form the mock response and reduce human error.
go test -v -run TestToBeConverted

In test output, you gain the output from original method in the way copy-pastable.

...
mm.Expect(&dep_pkg1.MethodA, "real response 1", "real response 2")
mm.Expect(&dep_pkg2.MethodB, "real response 3")
...
...
3. By copying them into your test, you gain the unittest version based on previous integration.
func TestToBeConverted(t *testing.T) {

    defer mm.Init(t).CleanUp()

    mm.Expect(&dep_pkg1.MethodA, "real response 1", "real response 2")
    mm.Expect(&dep_pkg2.MethodB, "real response 3")

    ... // Your original code here. No change.
}
4. After that, remove this line.
mm.SetLogger(mm.CapturingLogger)

Now you got a perfect unittest that fully detached from real backends.

How can I fake an object returned by the dependency pkg? (For refactoring scenarios)

Methodman is modeled around modifiable method stub, so it won't natively work for this kind of faking object.

When converting your implementation to allow Dependency Injection for this case, you probably will,

  1. Abstractise the returned type into an interface, which allow using a mock implementation behind

  2. Probably you'll use codegen tool like Testify Mock Mokery to generate the mock implementation in independent files

  3. By normal practice, you will need to change your main logic to receive dependency as extra param, to allow mocking in unittest.

However, for Step 3, Methodman makes it easier. You can just patch the function where you receive the object, either object constructor, or a singelton getter. Simply hijack it to response with your mock object, and that's all. No need to refactor main logic for enable mocking.

What about mocking an exported channel without significant refactoring? I'm not sure yet. Please share with me if you got idea.

Complete Demo

Please check out GitHub Pages

About

Methodman is a stub-based Denpendency Injection tool in Go

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages