@@ -10,21 +10,25 @@ import (
10
10
11
11
// NewAnalyzer returns a new analyzer to check for receiver type consistency.
12
12
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 {}{},
21
15
}
22
16
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
+ }
25
27
}
26
28
27
- a := & analyzer {excludedMethods : excludedMethods }
29
+ for _ , exclusion := range s .Exclusions {
30
+ a .excluded [exclusion ] = struct {}{}
31
+ }
28
32
29
33
return & analysis.Analyzer {
30
34
Name : "recvcheck" ,
@@ -45,10 +49,14 @@ type Settings struct {
45
49
// - "MarshalBinary"
46
50
// - "GobEncode"
47
51
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
48
56
}
49
57
50
58
type analyzer struct {
51
- excludedMethods map [string ]struct {}
59
+ excluded map [string ]struct {}
52
60
}
53
61
54
62
func (r * analyzer ) run (pass * analysis.Pass ) (any , error ) {
@@ -61,21 +69,12 @@ func (r *analyzer) run(pass *analysis.Pass) (any, error) {
61
69
return
62
70
}
63
71
64
- if r .isExcluded (funcDecl ) {
72
+ recv , isStar := recvTypeIdent (funcDecl .Recv .List [0 ].Type )
73
+ if recv == nil {
65
74
return
66
75
}
67
76
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 ) {
79
78
return
80
79
}
81
80
@@ -101,16 +100,36 @@ func (r *analyzer) run(pass *analysis.Pass) (any, error) {
101
100
return nil , nil
102
101
}
103
102
104
- func (r * analyzer ) isExcluded (f * ast.FuncDecl ) bool {
103
+ func (r * analyzer ) isExcluded (recv * ast. Ident , f * ast.FuncDecl ) bool {
105
104
if f .Name == nil || f .Name .Name == "" {
106
105
return true
107
106
}
108
107
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
+
110
115
return found
111
116
}
112
117
113
118
type structType struct {
114
119
starUsed bool
115
120
typeUsed bool
116
121
}
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
+ }
0 commit comments