Skip to content

Commit 2f6597c

Browse files
committedJan 30, 2023
Introduce GinkgoLabelFilter() and Label().MatchesLabelFilter() to make it possible to programatically match filters
1 parent ea4966e commit 2f6597c

File tree

8 files changed

+80
-1
lines changed

8 files changed

+80
-1
lines changed
 

‎core_dsl.go

+14
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,20 @@ func GinkgoHelper() {
172172
types.MarkAsHelper(1)
173173
}
174174

175+
/*
176+
GinkgoLabelFilter() returns the label filter configured for this suite via `--label-filter`.
177+
178+
You can use this to manually check if a set of labels would satisfy the filter via:
179+
180+
if (Label("cat", "dog").MatchesLabelFilter(GinkgoLabelFilter())) {
181+
//...
182+
}
183+
*/
184+
func GinkgoLabelFilter() string {
185+
suiteConfig, _ := GinkgoConfiguration()
186+
return suiteConfig.LabelFilter
187+
}
188+
175189
/*
176190
PauseOutputInterception() pauses Ginkgo's output interception. This is only relevant
177191
when running in parallel and output to stdout/stderr is being intercepted. You generally

‎docs/index.md

+15-1
Original file line numberDiff line numberDiff line change
@@ -1362,7 +1362,7 @@ DescribeTable("Extracting the author's first and last name",
13621362
You'll be notified with a clear message at runtime if the parameter types don't match the spec closure signature.
13631363

13641364
#### Mental Model: Table Specs are just Syntactic Sugar
1365-
`DescribeTable` is simply providing syntactic sugar to convert its inputs into a set of standard Ginkgo nodes. During the [Tree Construction Phase](#mental-model-how-ginkgo-traverses-the-spec-hierarchy) `DescribeTable` is generating a single container node that contains one subject node per table entry. The description for the container node will be the description passed to `DescribeTable` and the descriptions for the subject nodes will be the descriptions passed to the `Entry`s. During the Run Phase, when specs run, each subject node will simply invoke the spec closure passed to `DescribeTable`, passing in the parameters associated with the `Entry`.
1365+
`DescribeTable` is simply providing syntactic sugar to convert its Ls into a set of standard Ginkgo nodes. During the [Tree Construction Phase](#mental-model-how-ginkgo-traverses-the-spec-hierarchy) `DescribeTable` is generating a single container node that contains one subject node per table entry. The description for the container node will be the description passed to `DescribeTable` and the descriptions for the subject nodes will be the descriptions passed to the `Entry`s. During the Run Phase, when specs run, each subject node will simply invoke the spec closure passed to `DescribeTable`, passing in the parameters associated with the `Entry`.
13661366

13671367
To put it another way, the table test above is equivalent to:
13681368

@@ -2493,6 +2493,20 @@ You can list the labels used in a given package using the `ginkgo labels` subcom
24932493

24942494
You can iterate on different filters quickly with `ginkgo --dry-run -v --label-filter=FILTER`. This will cause Ginkgo to tell you which specs it will run for a given filter without actually running anything.
24952495

2496+
If you want to have finer-grained control within a test about what code to run/not-run depending on what labels match/don't match the filter you can perform a manual check against the label-filter passed into Ginkgo like so:
2497+
2498+
```go
2499+
It("can save books remotely", Label("network", "slow", "library query") {
2500+
if Label("performance").MatchesLabelFilter(GinkgoLabelFilter()) {
2501+
exp := gmeasure.NewExperiment()
2502+
// perform some benchmarking with exp...
2503+
}
2504+
// rest of the saving books test
2505+
})
2506+
```
2507+
2508+
here `GinkgoLabelFilter()` returns the configured label filter passed in via `--label-filter`. With a setup like this you could run `ginkgo --label-filter="network && !performance"` - this would select the `"can save books remotely"` spec but not run the benchmarking code in the spec. Of course, this could also have been modeled as a separate spec with the `performance` label.
2509+
24962510
Finally, in addition to specifying Labels on subject and container nodes you can also specify suite-wide labels by decorating the `RunSpecs` command with `Label`:
24972511

24982512
```go

‎dsl/core/core_dsl.go

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ var GinkgoConfiguration = ginkgo.GinkgoConfiguration
2929
var GinkgoRandomSeed = ginkgo.GinkgoRandomSeed
3030
var GinkgoParallelProcess = ginkgo.GinkgoParallelProcess
3131
var GinkgoHelper = ginkgo.GinkgoHelper
32+
var GinkgoLabelFilter = ginkgo.GinkgoLabelFilter
3233
var PauseOutputInterception = ginkgo.PauseOutputInterception
3334
var ResumeOutputInterception = ginkgo.ResumeOutputInterception
3435
var RunSpecs = ginkgo.RunSpecs

‎integration/_fixtures/filter_fixture/filter_suite_test.go

+5
Original file line numberDiff line numberDiff line change
@@ -11,3 +11,8 @@ func TestFilterFixture(t *testing.T) {
1111
RegisterFailHandler(Fail)
1212
RunSpecs(t, "FilterFixture Suite", Label("TopLevelLabel"))
1313
}
14+
15+
var _ = BeforeEach(func() {
16+
config, _ := GinkgoConfiguration()
17+
Ω(GinkgoLabelFilter()).Should(Equal(config.LabelFilter))
18+
})

‎internal/node.go

+4
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,10 @@ type NodeTimeout time.Duration
9393
type SpecTimeout time.Duration
9494
type GracePeriod time.Duration
9595

96+
func (l Labels) MatchesLabelFilter(query string) bool {
97+
return types.MustParseLabelFilter(query)(l)
98+
}
99+
96100
func UnionOfLabels(labels ...Labels) Labels {
97101
out := Labels{}
98102
seen := map[string]bool{}

‎internal/node_test.go

+16
Original file line numberDiff line numberDiff line change
@@ -1621,6 +1621,22 @@ var _ = Describe("Nodes", func() {
16211621
})
16221622

16231623
})
1624+
1625+
Describe("Labels", func() {
1626+
It("can match against a filter", func() {
1627+
Ω(Label().MatchesLabelFilter("")).Should(BeTrue())
1628+
Ω(Label("dog", "cat").MatchesLabelFilter("dog")).Should(BeTrue())
1629+
Ω(Label("dog", "cat").MatchesLabelFilter("cat")).Should(BeTrue())
1630+
Ω(Label("dog", "cat").MatchesLabelFilter("dog && cat")).Should(BeTrue())
1631+
Ω(Label("dog", "cat").MatchesLabelFilter("dog || cat")).Should(BeTrue())
1632+
Ω(Label("dog", "cat").MatchesLabelFilter("!fish")).Should(BeTrue())
1633+
Ω(Label("dog", "cat").MatchesLabelFilter("fish")).Should(BeFalse())
1634+
Ω(Label("dog", "cat").MatchesLabelFilter("!dog")).Should(BeFalse())
1635+
Ω(func() {
1636+
Label("dog", "cat").MatchesLabelFilter("!")
1637+
}).Should(Panic())
1638+
})
1639+
})
16241640
})
16251641

16261642
var _ = Describe("Iteration Performance", Serial, Label("performance"), func() {

‎types/label_filter.go

+11
Original file line numberDiff line numberDiff line change
@@ -272,12 +272,23 @@ func tokenize(input string) func() (*treeNode, error) {
272272
}
273273
}
274274

275+
func MustParseLabelFilter(input string) LabelFilter {
276+
filter, err := ParseLabelFilter(input)
277+
if err != nil {
278+
panic(err)
279+
}
280+
return filter
281+
}
282+
275283
func ParseLabelFilter(input string) (LabelFilter, error) {
276284
if DEBUG_LABEL_FILTER_PARSING {
277285
fmt.Println("\n==============")
278286
fmt.Println("Input: ", input)
279287
fmt.Print("Tokens: ")
280288
}
289+
if input == "" {
290+
return func(_ []string) bool { return true }, nil
291+
}
281292
nextToken := tokenize(input)
282293

283294
root := &treeNode{token: lfTokenRoot}

‎types/label_filter_test.go

+14
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,10 @@ var _ = Describe("LabelFilter", func() {
6868
}
6969
}
7070
},
71+
Entry("An empty label", "",
72+
M("cat"), M("cat", "dog"), M("dog", "cat"),
73+
M(), M("cow"),
74+
),
7175
Entry("A single label", "cat",
7276
M("cat"), M("cat", "dog"), M("dog", "cat"),
7377
NM(), NM("cow"),
@@ -187,4 +191,14 @@ var _ = Describe("LabelFilter", func() {
187191
Entry(nil, "cow)", "", types.GinkgoErrors.InvalidLabel("cow)", cl)),
188192
Entry(nil, "cow/", "", types.GinkgoErrors.InvalidLabel("cow/", cl)),
189193
)
194+
195+
Describe("MustParseLabelFilter", func() {
196+
It("panics if passed an invalid filter", func() {
197+
Ω(types.MustParseLabelFilter("dog")([]string{"dog"})).Should(BeTrue())
198+
Ω(types.MustParseLabelFilter("dog")([]string{"cat"})).Should(BeFalse())
199+
Ω(func() {
200+
types.MustParseLabelFilter("!")
201+
}).Should(Panic())
202+
})
203+
})
190204
})

0 commit comments

Comments
 (0)
Please sign in to comment.