This question: How to test os.exit scenarios in Go (and the highest voted answer therein) sets out how to test os.Exit()
scenarios within go. As os.Exit()
cannot easily be intercepted, the method used is to reinvoke the binary and check the exit value. This method is described at slide 23 on this presentation by Andrew Gerrand (one of the core members of the Go team); the code is very simple and is reproduced in full below.
The relevant test and main files look like this (note that this pair of files alone is an MVCE):
package foo
import (
"os"
"os/exec"
"testing"
)
func TestCrasher(t *testing.T) {
if os.Getenv("BE_CRASHER") == "1" {
Crasher() // This causes os.Exit(1) to be called
return
}
cmd := exec.Command(os.Args[0], "-test.run=TestCrasher")
cmd.Env = append(os.Environ(), "BE_CRASHER=1")
err := cmd.Run()
if e, ok := err.(*exec.ExitError); ok && !e.Success() {
fmt.Printf("Error is %v\n", e)
return
}
t.Fatalf("process ran with err %v, want exit status 1", err)
}
and
package foo
import (
"fmt"
"os"
)
// Coverage testing thinks (incorrectly) that the func below is
// never being called
func Crasher() {
fmt.Println("Going down in flames!")
os.Exit(1)
}
However, this method appears to suffer certain limitations:
Coverage testing with goveralls / coveralls.io does not work - see for instance the example here (the same code as above but put into github for your convenience) which produces the coverage test here, i.e. it does not record the test functions being run. NOTE that you don't need to those links to answer the question - the above example will work fine - they are just there to show what happens if you put the above into github, and take it all the way through travis to coveralls.io
Rerunning the test binary appears fragile.
Specifically, as requested, here is a screenshot (rather than a link) for the coverage failure; the red shading indicates that as far as coveralls.io is concerned, Crasher()
is not being called.
Is there a way around this? Particularly the first point.
At a golang level the problem is this:
The Goveralls framework runs
go test -cover ...
, which invokes the test above.The test above calls
exec.Command / .Run
without-cover
in the OS argumentsUnconditionally putting
-cover
etc. in the argument list is unattractive as it would then run a coverage test (as the subprocess) within a non-coverage test, and parsing the argument list for the presence of-cover
etc. seems a heavy duty solution.Even if I put
-cover
etc. in the argument list, my understanding is that I'd then have two coverage outputs written to the same file, which isn't going to work - these would need merging somehow. The closest I've got to that is this golang issue.
Summary
What I am after is a simple way to run go coverage testing (preferably via travis, goveralls, and coveralls.io), where it is possible to both test cases where the tested routine exits with OS.exit()
, and where the coverage of that test is noted. I'd quite like it to use the re-exec method above (if that can be made to work) if that can be made to work.
The solution should show coverage testing of Crasher()
. Excluding Crasher()
from coverage testing is not an option, as in the real world what I am trying to do is test a more complex function, where somewhere deep within, under certain conditions, it calls e.g. log.Fatalf()
; what I am coverage testing is that the tests for those conditions works properly.
With a slight refactoring, you may easily achieve 100% coverage.
foo/bar.go
:And the testing code:
foo/bar_test.go
:Running
go test -cover
:Yes, you might say this works if
os.Exit()
is called explicitly, but what ifos.Exit()
is called by someone else, e.g.log.Fatalf()
?The same technique works there too, you just have to switch
log.Fatalf()
instead ofos.Exit()
, e.g.:Relevant part of
foo/bar.go
:And the testing code:
TestCrasher()
infoo/bar_test.go
:Running
go test -cover
:Interfaces and mocks
Using Go interfaces possible to create mock-able compositions. A type could have interfaces as bound dependencies. These dependencies could be easily substituted with mocks appropriate to the interfaces.
Testing
Disadvantages
Original
Exit
method still won't be tested so it should be responsible only for exit, nothing more.Functions are first class citizens
Parameter dependency
Functions are first class citizens in Go. A lot of operations are allowed with functions so we can do some tricks with functions directly.
Using 'pass as parameter' operation we can do a dependency injection:
Testing:
Disadvantages
You must pass a dependency as a parameter. If you have many dependencies a length of params list could be huge.
Variable substitution
Actually it is possible to do it using "assign to variable" operation without explicit passing a function as a parameter.
Testing
disadvantages
It is implicit and easy to crash.
Design notes
If you plan to declare some logic below
Exit
an exit logic must be isolated withelse
block or extrareturn
after exit because mock won't stop execution.It not common practice to put tests around the
Main
function of an application inGOLANG
specifically because of issues like that. There was a question that is already answered that touched this same issue.To Summarize
To summarize it you should avoid putting tests around the main entry point of the application and try to design your application in way that little code is on the
Main
function so it decoupled enough to allow you to test as much of your code as possible.Check GOLANG Testing for more information.
Coverage to 100%
As I detailed on on the previous answer since is a bad idea to try getting tests around the
Main
func and the best practice is to put as little code there as possible so it can be tested properly with out blind spots it stands to reason that trying to get 100% coverage while trying to include theMain
func is wasted effort so it better to ignore it in the tests.You can use build tags to exclude the
main.go
file from the tests therefore reaching your 100% coverage or all green.Check: showing coverage of functional tests without blind spots
If you design your code well and keep all the actual functionality well decoupled and tested having a few lines of code that do very little other then calling the actual pieces of code that do all the actual work and are well tested it does't really matter that you are not testing a tiny and not significant code.