Skip to content

Commit a27aef1

Browse files
authored
feat: option to exclude specific methods (#11)
1 parent 04e3730 commit a27aef1

File tree

10 files changed

+86
-33
lines changed

10 files changed

+86
-33
lines changed

analyzer.go

+45-26
Original file line numberDiff line numberDiff line change
@@ -10,21 +10,25 @@ import (
1010

1111
// NewAnalyzer returns a new analyzer to check for receiver type consistency.
1212
func NewAnalyzer(s Settings) *analysis.Analyzer {
13-
// Default excludes for Marshal/Encode methods https://github.com/raeperd/recvcheck/issues/7
14-
excludedMethods := map[string]struct{}{
15-
"MarshalText": {},
16-
"MarshalJSON": {},
17-
"MarshalYAML": {},
18-
"MarshalXML": {},
19-
"MarshalBinary": {},
20-
"GobEncode": {},
13+
a := &analyzer{
14+
excluded: map[string]struct{}{},
2115
}
2216

23-
if s.DisableBuiltin {
24-
excludedMethods = map[string]struct{}{}
17+
if !s.DisableBuiltin {
18+
// Default excludes for Marshal/Encode methods https://github.com/raeperd/recvcheck/issues/7
19+
a.excluded = map[string]struct{}{
20+
"*.MarshalText": {},
21+
"*.MarshalJSON": {},
22+
"*.MarshalYAML": {},
23+
"*.MarshalXML": {},
24+
"*.MarshalBinary": {},
25+
"*.GobEncode": {},
26+
}
2527
}
2628

27-
a := &analyzer{excludedMethods: excludedMethods}
29+
for _, exclusion := range s.Exclusions {
30+
a.excluded[exclusion] = struct{}{}
31+
}
2832

2933
return &analysis.Analyzer{
3034
Name: "recvcheck",
@@ -45,10 +49,14 @@ type Settings struct {
4549
// - "MarshalBinary"
4650
// - "GobEncode"
4751
DisableBuiltin bool
52+
53+
// Exclusions format is `struct_name.method_name` (ex: `Foo.MethodName`).
54+
// A wildcard `*` can use as a struct name (ex: `*.MethodName`).
55+
Exclusions []string
4856
}
4957

5058
type analyzer struct {
51-
excludedMethods map[string]struct{}
59+
excluded map[string]struct{}
5260
}
5361

5462
func (r *analyzer) run(pass *analysis.Pass) (any, error) {
@@ -61,21 +69,12 @@ func (r *analyzer) run(pass *analysis.Pass) (any, error) {
6169
return
6270
}
6371

64-
if r.isExcluded(funcDecl) {
72+
recv, isStar := recvTypeIdent(funcDecl.Recv.List[0].Type)
73+
if recv == nil {
6574
return
6675
}
6776

68-
var recv *ast.Ident
69-
var isStar bool
70-
switch recvType := funcDecl.Recv.List[0].Type.(type) {
71-
case *ast.StarExpr:
72-
isStar = true
73-
if recv, ok = recvType.X.(*ast.Ident); !ok {
74-
return
75-
}
76-
case *ast.Ident:
77-
recv = recvType
78-
default:
77+
if r.isExcluded(recv, funcDecl) {
7978
return
8079
}
8180

@@ -101,16 +100,36 @@ func (r *analyzer) run(pass *analysis.Pass) (any, error) {
101100
return nil, nil
102101
}
103102

104-
func (r *analyzer) isExcluded(f *ast.FuncDecl) bool {
103+
func (r *analyzer) isExcluded(recv *ast.Ident, f *ast.FuncDecl) bool {
105104
if f.Name == nil || f.Name.Name == "" {
106105
return true
107106
}
108107

109-
_, found := r.excludedMethods[f.Name.Name]
108+
_, found := r.excluded[recv.Name+"."+f.Name.Name]
109+
if found {
110+
return true
111+
}
112+
113+
_, found = r.excluded["*."+f.Name.Name]
114+
110115
return found
111116
}
112117

113118
type structType struct {
114119
starUsed bool
115120
typeUsed bool
116121
}
122+
123+
func recvTypeIdent(r ast.Expr) (*ast.Ident, bool) {
124+
switch n := r.(type) {
125+
case *ast.StarExpr:
126+
if i, ok := n.X.(*ast.Ident); ok {
127+
return i, true
128+
}
129+
130+
case *ast.Ident:
131+
return n, false
132+
}
133+
134+
return nil, false
135+
}

analyzer_test.go

+9-1
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,21 @@ func TestAnalyzer(t *testing.T) {
1717
settings: recvcheck.Settings{},
1818
},
1919
{
20-
desc: "excluded",
20+
desc: "builtinmethods",
2121
settings: recvcheck.Settings{},
2222
},
2323
{
2424
desc: "disablebuiltin",
2525
settings: recvcheck.Settings{DisableBuiltin: true},
2626
},
27+
{
28+
desc: "exclusions",
29+
settings: recvcheck.Settings{Exclusions: []string{"SQL.Value"}},
30+
},
31+
{
32+
desc: "exclusionswildcard",
33+
settings: recvcheck.Settings{Exclusions: []string{"*.Value"}},
34+
},
2735
}
2836

2937
for _, test := range testCases {

testdata/src/excluded/binary.go testdata/src/builtinmethods/binary.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
type Binary struct{}
44

testdata/src/excluded/gob.go testdata/src/builtinmethods/gob.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
type Gob struct{}
44

testdata/src/excluded/json.go testdata/src/builtinmethods/json.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
type JSON struct{}
44

testdata/src/excluded/text.go testdata/src/builtinmethods/text.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
type Text struct{}
44

testdata/src/excluded/xml.go testdata/src/builtinmethods/xml.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
import "encoding/xml"
44

testdata/src/excluded/yaml.go testdata/src/builtinmethods/yaml.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package excluded
1+
package builtinmethods
22

33
type Node struct{}
44

testdata/src/exclusions/exclusions.go

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package exclusions
2+
3+
import "database/sql/driver"
4+
5+
type SQL struct{}
6+
7+
func (s SQL) Value() (driver.Value, error) {
8+
panic("not implemented")
9+
}
10+
11+
func (s *SQL) Scan(src any) error {
12+
panic("not implemented")
13+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package exclusions
2+
3+
import "database/sql/driver"
4+
5+
type SQL struct{}
6+
7+
func (s SQL) Value() (driver.Value, error) {
8+
panic("not implemented")
9+
}
10+
11+
func (s *SQL) Scan(src any) error {
12+
panic("not implemented")
13+
}

0 commit comments

Comments
 (0)