-
Notifications
You must be signed in to change notification settings - Fork 17.9k
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
proposal: testing: per-subtest setup and cleanup #59291
Comments
You can get a similar effect today by writing code like func TestCounter(t *testing.T) {
var a CoffeeMachine
wrapper := func(f func(t *testing.T)) func(t *testing.T) {
return func(t *testing.T) {
a = CoffeeMachine{}
f(t)
}
}
t.Run("on", wrapper(func(t *testing.T) {
err := a.TurnOn()
...
}))
} You can ensure that There is probably room for more control over tests and subtests, but in my opinion this proposal is too specific, adding functionality that will rarely be used. |
@ianlancetaylor, yes, I considered a wrapper like yours. But in fact it does not resolve the same points against direct setup in subtests: you can forget to wrap and you can wrap at wrong level. |
Closed by mistake, sorry |
It seems like you could equally forget to use BeforeRun. Can you demonstrate what you mean by wrap at the wrong level? |
Once you set BeforeRun, you can be sure that it will be called for a new added subtests of the same parent.
I meant to call wrap at other level of subtests than the level where it’s defined (deeper than necessary) |
@willfaught I think your objection could also apply to |
The functionality provided by the testing package today makes it quite easy to accomplish this style of testing. Here's an another example similar to the one above, which I think may do a better job of addressing the problems identified in the PR description. func TestCoffeeMachine(t *testing.T) {
givenCoffeeMachine := func(fn func(*testing.T, CoffeeMachine)) {
m := CoffeeMachine{}
t.Run("given a new coffee machine", func(t *testing.T) {
// t.Cleanup( ... ) can be done here
fn(t, m)
})
}
givenCoffeeMachine(func(t *testing.T, a CoffeeMachine) {
t.Run("it could be turned on", func(t *testing.T) {
err := a.TurnOn()
assert.NilError(t, err)
t.Run("it could not be turned on again", func(t *testing.T) {
err := a.TurnOn()
assert.Error(t, err, "already on")
})
t.Run("it makes a coffee", func(t *testing.T) {
err := a.MakeCoffee()
assert.NilError(t, err)
})
})
})
givenCoffeeMachine(func(t *testing.T, a CoffeeMachine) {
t.Run("MakeCoffee fails because it's not turned on", func(t *testing.T) {
err := a.MakeCoffee()
assert.Error(t, err, "power is off")
})
})
} This approach avoids the shared |
What can be accomplished with |
I was playing around with the suggestion provided by @dnephin above. Here's my example:
(edited to use defer for cleanup instead of t.Cleanup) While the above works just fine, it also is a pattern that needs to be copied around a codebase and might drift overtime. Having a solution in the standard library would mostly prevent drift. I've also been working on a small library that wraps |
When writing tests in BDD style using subtests it's often necessary to invoke some common code before or after a subtest. In the example below we call
beforeRun
in the first place of subtests 1 and 2, otherwise the subtests would affect the outcome of each other:It would be nice instead, if the testing package could invoke a specific function by itself before/after direct subtests of the parent where it's configured:
The points against of the explicit
beforeRun(t)
calls from subtests are the following:So I propose to add
func (t *T) BeforeRun(func(t *testing.T)
andfunc (t *T) AfterRun(func(t *testing.T)
methods which should be invoked by the testing package before/after direct subtests of the parent where it's configured.Relates to #39222 #40984
The text was updated successfully, but these errors were encountered: