-
Notifications
You must be signed in to change notification settings - Fork 128
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support running tests using RR #365
Comments
I updated the description to reflect the fact that go-test-rr.gopackage main
import (
"fmt"
"os"
"os/exec"
"path/filepath"
"strings"
)
func fatalf(format string, args ...interface{}) {
fmt.Fprintf(os.Stderr, "Error: "+format+"\n", args...)
os.Exit(1)
}
func check(err error) {
if err != nil {
fatalf("%v", err)
}
}
func main() {
// Make the trace directory
check(os.MkdirAll("trace", 0700))
// List all packages that contain test files
cmd := exec.Command("go", "list", "-f", "{{if .TestGoFiles}}{{.ImportPath}}{{end}}", "./...")
out, err := cmd.CombinedOutput()
check(err)
// Test each one
for _, pkg := range strings.Split(string(out), "\n") {
pkg = strings.TrimSpace(pkg)
pkg = "." + strings.TrimPrefix(pkg, "gitlab.com/accumulatenetwork/accumulate")
run(pkg, os.Args[1:])
}
}
func run(pkg string, args []string) {
// Clear the trace directory
cwd, err := os.Getwd()
check(err)
dir := filepath.Join(cwd, "trace", pkg)
check(os.RemoveAll(dir))
check(os.MkdirAll(filepath.Dir(dir), 0700))
// Compile the binary
cmd := exec.Command("go", append([]string{"test", "-json", "-gcflags", "all=-N -l", "-exec", "rr record -o " + dir, pkg}, args...)...)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
check(cmd.Run())
} |
Thank you for suggesting this feature! How would you generally use this? Would you run the entire test suite of a project then use Would you use it in CI and save the trace files as artifacts? I see under limitations that it emulates a single core, so I imagine if you did run it in CI you'd want to do that as a completely separate job and not the primary CI test run. Any interesting CI failures could probably be reproduced locally by rerunning that package individually a large number of times. That might be easier than duplicating the test runs, and saving many trace artifacts, for the rare chance you need the debugger. Maybe this could be integrated with a In both cases saving the file under the project directory sounds great. We could include a date and time in the filename instead of deleting it each time. That would produce a lot of files. Is there any reason to keep record files for successful runs? If we only keep the traces for a package with test failures that might help mitigate the problem of too many files. If someone were trying to trace down a flaky behaviour, they could run the tests in a loop many times and end up with only a few files for the failed runs. Maybe a How would that work for your usage or |
My motivation is flaky tests that rarely fail when run on a developer’s computer but sporadically fail in CI. I’ve had these kinds of failures show up periodically and they’re a huge pain to debug because they’re often almost impossible to reproduce reliably. When one of these tests fail, I want to download the trace and step through it to debug the failure.
The way that limitation is phrased is a bit misleading I think. I also made the assumption that rr would be useless for reproducing concurrency issues. A better way of putting it is that only one instruction is running at any given time, but the code can still be concurrent. It is true that I’ve had trouble reproducing concurrency issues when running in normal mode. However rr also has ‘chaos mode’ which is very helpful for reproducing races.
I only have a use for records of failed runs. I could invent a reason for using record of a successful run but I think in the end that would essentially mean a test was succeeding when it really should be failing. What you suggest would work perfectly for me, especially with a flag for chaos mode (or just a pass through -rrflags). It would be particularly helpful if I could selectively enable chaos mode for specific packages (config file?). I’m working on distributed systems and I think my simulator has numerous concurrency bugs, so being able to selectively enable chaos mode would allow me to work on those bugs incrementally. |
It would be amazing if gotestsum supported running tests using rr. I've hacked something together to use with
--raw-command
but it's ugly. I am imagining something likegotestsum --engine rr
(with the default being--engine go
)./...
go list -f '{{if .TestGoFiles}}{{.ImportPath}}{{end}}' $1
to list only the packages containing test files-exec 'rr record -o $TRACE'
to thego test
invocationpkg/foo/bar
will be written to~/.local/share/rr/bar.test-0
The text was updated successfully, but these errors were encountered: