Skip to content

Commit ed149aa

Browse files
committed
added errgroupcheck linter
1 parent 78a738f commit ed149aa

12 files changed

+187
-0
lines changed

.golangci.next.reference.yml

+5
Original file line numberDiff line numberDiff line change
@@ -324,6 +324,11 @@ linters-settings:
324324
# Default: false
325325
report-no-exported: false
326326

327+
errgroupcheck:
328+
# Check if any sync.errgroup.Group instance is missing a call to the Wait() func.
329+
# Default: true
330+
require-wait: true
331+
327332
errorlint:
328333
# Check whether fmt.Errorf uses the %w verb for formatting errors.
329334
# See the https://github.com/polyfloyd/go-errorlint for caveats.

go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ require (
1616
github.com/GaijinEntertainment/go-exhaustruct/v3 v3.3.0
1717
github.com/OpenPeeDeeP/depguard/v2 v2.2.0
1818
github.com/alecthomas/go-check-sumtype v0.1.4
19+
github.com/alexbagnolini/errgroupcheck v0.1.2
1920
github.com/alexkohler/nakedret/v2 v2.0.4
2021
github.com/alexkohler/prealloc v1.0.0
2122
github.com/alingse/asasalint v0.0.11

go.sum

+2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

jsonschema/golangci.next.jsonschema.json

+12
Original file line numberDiff line numberDiff line change
@@ -312,6 +312,7 @@
312312
"durationcheck",
313313
"errcheck",
314314
"errchkjson",
315+
"errgroupcheck",
315316
"errname",
316317
"errorlint",
317318
"execinquery",
@@ -865,6 +866,17 @@
865866
}
866867
}
867868
},
869+
"errgroupcheck": {
870+
"type": "object",
871+
"additionalProperties": false,
872+
"properties": {
873+
"require-wait": {
874+
"description": "Check if any sync.errgroup.Group instance is missing a call to the Wait() func.",
875+
"type": "boolean",
876+
"default": true
877+
}
878+
}
879+
},
868880
"errorlint": {
869881
"type": "object",
870882
"additionalProperties": false,

pkg/config/linters_settings.go

+5
Original file line numberDiff line numberDiff line change
@@ -208,6 +208,7 @@ type LintersSettings struct {
208208
DupWord DupWordSettings
209209
Errcheck ErrcheckSettings
210210
ErrChkJSON ErrChkJSONSettings
211+
ErrGroupCheck ErrGroupCheckSettings
211212
ErrorLint ErrorLintSettings
212213
Exhaustive ExhaustiveSettings
213214
Exhaustruct ExhaustructSettings
@@ -382,6 +383,10 @@ type ErrChkJSONSettings struct {
382383
ReportNoExported bool `mapstructure:"report-no-exported"`
383384
}
384385

386+
type ErrGroupCheckSettings struct {
387+
RequireWait bool `mapstructure:"require-wait"`
388+
}
389+
385390
type ErrorLintSettings struct {
386391
Errorf bool `mapstructure:"errorf"`
387392
ErrorfMulti bool `mapstructure:"errorf-multi"`
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
package errgroupcheck
2+
3+
import (
4+
"github.com/alexbagnolini/errgroupcheck"
5+
"github.com/golangci/golangci-lint/pkg/config"
6+
"github.com/golangci/golangci-lint/pkg/goanalysis"
7+
"golang.org/x/tools/go/analysis"
8+
)
9+
10+
func New(cfg *config.ErrGroupCheckSettings) *goanalysis.Linter {
11+
var setts = errgroupcheck.DefaultSettings()
12+
13+
if cfg != nil {
14+
setts.RequireWait = cfg.RequireWait
15+
}
16+
17+
cfgMap := map[string]map[string]any{}
18+
19+
analyzer := errgroupcheck.NewAnalyzer(setts)
20+
21+
if cfg != nil {
22+
cfgMap[analyzer.Name] = map[string]any{
23+
"require-wait": cfg.RequireWait,
24+
}
25+
}
26+
27+
return goanalysis.NewLinter(
28+
analyzer.Name,
29+
analyzer.Doc,
30+
[]*analysis.Analyzer{analyzer},
31+
nil,
32+
).WithLoadMode(goanalysis.LoadModeTypesInfo)
33+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
package errgroupcheck
2+
3+
import (
4+
"testing"
5+
6+
"github.com/golangci/golangci-lint/test/testshared/integration"
7+
)
8+
9+
func TestFromTestdata(t *testing.T) {
10+
integration.RunTestdata(t)
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
//golangcitest:args -Eerrgroupcheck
2+
//golangcitest:config_path testdata/errgroupcheck_wait.yml
3+
package testdata
4+
5+
import (
6+
"context"
7+
8+
"golang.org/x/sync/errgroup"
9+
)
10+
11+
func ErrgroupWithWait() {
12+
eg := errgroup.Group{}
13+
14+
eg.Go(func() error {
15+
return nil
16+
})
17+
18+
eg.Go(func() error {
19+
return nil
20+
})
21+
22+
_ = eg.Wait()
23+
}
24+
25+
func ErrgroupMissingWait() {
26+
eg := errgroup.Group{} // want "errgroup 'eg' does not have Wait called"
27+
28+
eg.Go(func() error {
29+
return nil
30+
})
31+
32+
eg.Go(func() error {
33+
return nil
34+
})
35+
}
36+
37+
func ErrgroupContextWithWait() {
38+
eg, _ := errgroup.WithContext(context.Background())
39+
40+
eg.Go(func() error {
41+
return nil
42+
})
43+
44+
eg.Go(func() error {
45+
return nil
46+
})
47+
48+
_ = eg.Wait()
49+
}
50+
51+
func ErrgroupContextMissingWait() {
52+
eg, _ := errgroup.WithContext(context.Background()) // want "errgroup 'eg' does not have Wait called"
53+
54+
eg.Go(func() error {
55+
return nil
56+
})
57+
58+
eg.Go(func() error {
59+
return nil
60+
})
61+
}
62+
63+
func ErrgroupMultipleScopesWithWait() {
64+
eg := errgroup.Group{}
65+
66+
eg.Go(func() error {
67+
return nil
68+
})
69+
70+
eg.Go(func() error {
71+
eg2 := errgroup.Group{}
72+
73+
eg2.Go(func() error {
74+
return nil
75+
})
76+
77+
return eg2.Wait()
78+
})
79+
80+
_ = eg.Wait()
81+
}
82+
83+
func ErrgroupMultipleScopesMissingWait() {
84+
eg := errgroup.Group{}
85+
86+
eg.Go(func() error {
87+
return nil
88+
})
89+
90+
eg.Go(func() error {
91+
eg2 := errgroup.Group{} // want "errgroup 'eg2' does not have Wait called"
92+
93+
eg2.Go(func() error {
94+
return nil
95+
})
96+
97+
return nil
98+
})
99+
100+
_ = eg.Wait()
101+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
linters-settings:
2+
errgroupcheck:
3+
require-wait: true
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
module wait
2+
3+
go 1.19
4+
5+
require golang.org/x/sync v0.7.0

pkg/golinters/errgroupcheck/testdata/go.sum

+2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

pkg/lint/lintersdb/builder_linter.go

+7
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import (
2121
"github.com/golangci/golangci-lint/pkg/golinters/err113"
2222
"github.com/golangci/golangci-lint/pkg/golinters/errcheck"
2323
"github.com/golangci/golangci-lint/pkg/golinters/errchkjson"
24+
"github.com/golangci/golangci-lint/pkg/golinters/errgroupcheck"
2425
"github.com/golangci/golangci-lint/pkg/golinters/errname"
2526
"github.com/golangci/golangci-lint/pkg/golinters/errorlint"
2627
"github.com/golangci/golangci-lint/pkg/golinters/execinquery"
@@ -236,6 +237,12 @@ func (LinterBuilder) Build(cfg *config.Config) ([]*linter.Config, error) {
236237
WithLoadForGoAnalysis().
237238
WithURL("https://github.com/breml/errchkjson"),
238239

240+
linter.NewConfig(errgroupcheck.New(&cfg.LintersSettings.ErrGroupCheck)).
241+
WithSince("1.60.0").
242+
WithPresets(linter.PresetBugs).
243+
WithLoadForGoAnalysis().
244+
WithURL("https://github.com/alexbagnolini/errgroupcheck"),
245+
239246
linter.NewConfig(errname.New()).
240247
WithSince("v1.42.0").
241248
WithPresets(linter.PresetStyle).

0 commit comments

Comments
 (0)