mirror of
https://github.com/mgechev/revive.git
synced 2025-06-12 23:37:32 +02:00
feature: add support for omitzero in rule struct-tag (#1238)
This commit is contained in:
parent
a59a2288fd
commit
8ece20b078
@ -38,6 +38,7 @@ var (
|
|||||||
go115 = goversion.Must(goversion.NewVersion("1.15"))
|
go115 = goversion.Must(goversion.NewVersion("1.15"))
|
||||||
go121 = goversion.Must(goversion.NewVersion("1.21"))
|
go121 = goversion.Must(goversion.NewVersion("1.21"))
|
||||||
go122 = goversion.Must(goversion.NewVersion("1.22"))
|
go122 = goversion.Must(goversion.NewVersion("1.22"))
|
||||||
|
go124 = goversion.Must(goversion.NewVersion("1.24"))
|
||||||
)
|
)
|
||||||
|
|
||||||
// Files return package's files.
|
// Files return package's files.
|
||||||
@ -210,6 +211,11 @@ func (p *Package) IsAtLeastGo122() bool {
|
|||||||
return p.goVersion.GreaterThanOrEqual(go122)
|
return p.goVersion.GreaterThanOrEqual(go122)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsAtLeastGo124 returns true if the Go version for this package is 1.24 or higher, false otherwise
|
||||||
|
func (p *Package) IsAtLeastGo124() bool {
|
||||||
|
return p.goVersion.GreaterThanOrEqual(go124)
|
||||||
|
}
|
||||||
|
|
||||||
func getSortableMethodFlagForFunction(fn *ast.FuncDecl) sortableMethodsFlags {
|
func getSortableMethodFlagForFunction(fn *ast.FuncDecl) sortableMethodsFlags {
|
||||||
switch {
|
switch {
|
||||||
case astutils.FuncSignatureIs(fn, "Len", []string{}, []string{"int"}):
|
case astutils.FuncSignatureIs(fn, "Len", []string{}, []string{"int"}):
|
||||||
|
@ -56,6 +56,7 @@ func (r *StructTagRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure
|
|||||||
w := lintStructTagRule{
|
w := lintStructTagRule{
|
||||||
onFailure: onFailure,
|
onFailure: onFailure,
|
||||||
userDefined: r.userDefined,
|
userDefined: r.userDefined,
|
||||||
|
isAtLeastGo124: file.Pkg.IsAtLeastGo124(),
|
||||||
}
|
}
|
||||||
|
|
||||||
ast.Walk(w, file.AST)
|
ast.Walk(w, file.AST)
|
||||||
@ -73,6 +74,7 @@ type lintStructTagRule struct {
|
|||||||
userDefined map[string][]string // map: key -> []option
|
userDefined map[string][]string // map: key -> []option
|
||||||
usedTagNbr map[int]bool // list of used tag numbers
|
usedTagNbr map[int]bool // list of used tag numbers
|
||||||
usedTagName map[string]bool // list of used tag keys
|
usedTagName map[string]bool // list of used tag keys
|
||||||
|
isAtLeastGo124 bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w lintStructTagRule) Visit(node ast.Node) ast.Visitor {
|
func (w lintStructTagRule) Visit(node ast.Node) ast.Visitor {
|
||||||
@ -281,6 +283,11 @@ func (w lintStructTagRule) checkJSONTag(name string, options []string) (string,
|
|||||||
if name != "-" {
|
if name != "-" {
|
||||||
return "option can not be empty in JSON tag", false
|
return "option can not be empty in JSON tag", false
|
||||||
}
|
}
|
||||||
|
case "omitzero":
|
||||||
|
if w.isAtLeastGo124 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
fallthrough
|
||||||
default:
|
default:
|
||||||
if w.isUserDefined(keyJSON, opt) {
|
if w.isUserDefined(keyJSON, opt) {
|
||||||
continue
|
continue
|
||||||
|
@ -16,3 +16,7 @@ func TestStructTagWithUserOptions(t *testing.T) {
|
|||||||
Arguments: []any{"json,inline,outline", "bson,gnu"},
|
Arguments: []any{"json,inline,outline", "bson,gnu"},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestStructTagAfterGo1_24(t *testing.T) {
|
||||||
|
testRule(t, "go1.24/struct_tag", &rule.StructTagRule{})
|
||||||
|
}
|
||||||
|
3
testdata/go1.24/go.mod
vendored
Normal file
3
testdata/go1.24/go.mod
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
module github.com/mgechev/revive/testdata
|
||||||
|
|
||||||
|
go 1.24
|
125
testdata/go1.24/struct_tag.go
vendored
Normal file
125
testdata/go1.24/struct_tag.go
vendored
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
package fixtures
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
type decodeAndValidateRequest struct {
|
||||||
|
// BEAWRE : the flag of URLParam should match the const string URLParam
|
||||||
|
URLParam string `json:"-" path:"url_param" validate:"numeric"`
|
||||||
|
Text string `json:"text" validate:"max=10"`
|
||||||
|
DefaultInt int `json:"defaultInt" default:"10.0"` // MATCH /field's type and default value's type mismatch/
|
||||||
|
DefaultInt2 int `json:"defaultInt2" default:"10"`
|
||||||
|
// MATCH:12 /unknown option 'inline' in JSON tag/
|
||||||
|
DefaultInt3 int `json:"defaultInt2,inline" default:"11"` // MATCH /duplicate tag name: 'defaultInt2'/
|
||||||
|
DefaultString string `json:"defaultString" default:"foo"`
|
||||||
|
DefaultBool bool `json:"defaultBool" default:"trues"` // MATCH /field's type and default value's type mismatch/
|
||||||
|
DefaultBool2 bool `json:"defaultBool2" default:"true"`
|
||||||
|
DefaultBool3 bool `json:"defaultBool3" default:"false"`
|
||||||
|
DefaultFloat float64 `json:"defaultFloat" default:"f10.0"` // MATCH /field's type and default value's type mismatch/
|
||||||
|
DefaultFloat2 float64 `json:"defaultFloat2" default:"10.0"`
|
||||||
|
MandatoryStruct mandatoryStruct `json:"mandatoryStruct" required:"trues"` // MATCH /required should be 'true' or 'false'/
|
||||||
|
MandatoryStruct2 mandatoryStruct `json:"mandatoryStruct2" required:"true"`
|
||||||
|
MandatoryStruct4 mandatoryStruct `json:"mandatoryStruct4" required:"false"`
|
||||||
|
OptionalStruct *optionalStruct `json:"optionalStruct,omitempty"`
|
||||||
|
OptionalQuery string `json:"-" querystring:"queryfoo"`
|
||||||
|
optionalQuery string `json:"-" querystring:"queryfoo"` // MATCH /tag on not-exported field optionalQuery/
|
||||||
|
// No-reg test for bug https://github.com/mgechev/revive/issues/208
|
||||||
|
Tiret string `json:"-,"`
|
||||||
|
BadTiret string `json:"other,"` // MATCH /option can not be empty in JSON tag/
|
||||||
|
ForOmitzero string `json:"forOmitZero,omitzero"` // 'omitzero' is valid in go 1.24
|
||||||
|
}
|
||||||
|
|
||||||
|
type RangeAllocation struct {
|
||||||
|
metav1.TypeMeta `json:",inline"` // MATCH /unknown option 'inline' in JSON tag/
|
||||||
|
metav1.ObjectMeta `json:"metadata,omitempty"`
|
||||||
|
Range string `json:"range,flow"` // MATCH /unknown option 'flow' in JSON tag/
|
||||||
|
Data []byte `json:"data,inline"` // MATCH /unknown option 'inline' in JSON tag/
|
||||||
|
}
|
||||||
|
|
||||||
|
type RangeAllocation struct {
|
||||||
|
metav1.TypeMeta `bson:",minsize"`
|
||||||
|
metav1.ObjectMeta `bson:"metadata,omitempty"`
|
||||||
|
Range string `bson:"range,flow"` // MATCH /unknown option 'flow' in BSON tag/
|
||||||
|
Data []byte `bson:"data,inline"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestContextSpecificTags2 struct {
|
||||||
|
A int `asn1:"explicit,tag:1"`
|
||||||
|
B int `asn1:"tag:2"`
|
||||||
|
S string `asn1:"tag:0,utf8"`
|
||||||
|
Ints []int `asn1:"set"`
|
||||||
|
Version int `asn1:"optional,explicit,default:0,tag:000"` // MATCH /duplicated tag number 0/
|
||||||
|
Time time.Time `asn1:"explicit,tag:4,other"` // MATCH /unknown option 'other' in ASN1 tag/
|
||||||
|
X int `asn1:"explicit,tag:invalid"` // MATCH /ASN1 tag must be a number, got 'invalid'/
|
||||||
|
}
|
||||||
|
|
||||||
|
type VirtualMachineRelocateSpecDiskLocator struct {
|
||||||
|
DynamicData
|
||||||
|
|
||||||
|
DiskId int32 `xml:"diskId,attr,cdata"`
|
||||||
|
Datastore ManagedObjectReference `xml:"datastore,chardata,innerxml"`
|
||||||
|
DiskMoveType string `xml:"diskMoveType,omitempty,comment"`
|
||||||
|
DiskBackingInfo BaseVirtualDeviceBackingInfo `xml:"diskBackingInfo,omitempty,any"`
|
||||||
|
Profile []BaseVirtualMachineProfileSpec `xml:"profile,omitempty,other"` // MATCH /unknown option 'other' in XML tag/
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestDuplicatedXMLTags struct {
|
||||||
|
A int `xml:"a"`
|
||||||
|
B int `xml:"a"` // MATCH /duplicate tag name: 'a'/
|
||||||
|
C int `xml:"c"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestDuplicatedBSONTags struct {
|
||||||
|
A int `bson:"b"`
|
||||||
|
B int `bson:"b"` // MATCH /duplicate tag name: 'b'/
|
||||||
|
C int `bson:"c"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestDuplicatedYAMLTags struct {
|
||||||
|
A int `yaml:"b"`
|
||||||
|
B int `yaml:"c"`
|
||||||
|
C int `yaml:"c"` // MATCH /duplicate tag name: 'c'/
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestDuplicatedProtobufTags struct {
|
||||||
|
A int `protobuf:"varint,name=b"`
|
||||||
|
B int `protobuf:"varint,name=c"`
|
||||||
|
C int `protobuf:"varint,name=c"` // MATCH /duplicate tag name: 'c'/
|
||||||
|
}
|
||||||
|
|
||||||
|
// test case from
|
||||||
|
// sigs.k8s.io/kustomize/api/types/helmchartargs.go
|
||||||
|
|
||||||
|
type HelmChartArgs struct {
|
||||||
|
ChartName string `json:"chartName,omitempty" yaml:"chartName,omitempty"`
|
||||||
|
ChartVersion string `json:"chartVersion,omitempty" yaml:"chartVersion,omitempty"`
|
||||||
|
ChartRepoURL string `json:"chartRepoUrl,omitempty" yaml:"chartRepoUrl,omitempty"`
|
||||||
|
ChartHome string `json:"chartHome,omitempty" yaml:"chartHome,omitempty"`
|
||||||
|
ChartRepoName string `json:"chartRepoName,omitempty" yaml:"chartRepoName,omitempty"`
|
||||||
|
HelmBin string `json:"helmBin,omitempty" yaml:"helmBin,omitempty"`
|
||||||
|
HelmHome string `json:"helmHome,omitempty" yaml:"helmHome,omitempty"`
|
||||||
|
Values string `json:"values,omitempty" yaml:"values,omitempty"`
|
||||||
|
ValuesLocal map[string]interface{} `json:"valuesLocal,omitempty" yaml:"valuesLocal,omitempty"`
|
||||||
|
ValuesMerge string `json:"valuesMerge,omitempty" yaml:"valuesMerge,omitempty"`
|
||||||
|
ReleaseName string `json:"releaseName,omitempty" yaml:"releaseName,omitempty"`
|
||||||
|
ReleaseNamespace string `json:"releaseNamespace,omitempty" yaml:"releaseNamespace,omitempty"`
|
||||||
|
ExtraArgs []string `json:"extraArgs,omitempty" yaml:"extraArgs,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test message for holding primitive types.
|
||||||
|
type Simple struct {
|
||||||
|
OBool *bool `protobuf:"varint,1,req,json=oBool"` // MATCH /protobuf tag lacks mandatory option 'name'/
|
||||||
|
OInt32 *int32 `protobuf:"varint,2,opt,name=o_int32,jsonx=oInt32"` // MATCH /unknown option 'jsonx' in protobuf tag/
|
||||||
|
OInt32Str *int32 `protobuf:"varint,3,rep,name=o_int32_str,name=oInt32Str"` // MATCH /protobuf tag has duplicated option 'name'/
|
||||||
|
OInt64 *int64 `protobuf:"varint,4,opt,json=oInt64,name=o_int64,json=oInt64"` // MATCH /protobuf tag has duplicated option 'json'/
|
||||||
|
OSint32Str *int32 `protobuf:"zigzag32,11,opt,name=o_sint32_str,json=oSint32Str"`
|
||||||
|
OSint64Str *int64 `protobuf:"zigzag64,13,opt,name=o_sint32_str,json=oSint64Str"` // MATCH /duplicate tag name: 'o_sint32_str'/
|
||||||
|
OFloat *float32 `protobuf:"fixed32,14,opt,name=o_float,json=oFloat"`
|
||||||
|
ODouble *float64 `protobuf:"fixed64,014,opt,name=o_double,json=oDouble"` // MATCH /duplicated tag number 14/
|
||||||
|
ODoubleStr *float64 `protobuf:"fixed6,17,opt,name=o_double_str,json=oDoubleStr"` // MATCH /invalid protobuf tag name 'fixed6'/
|
||||||
|
OString *string `protobuf:"bytes,18,opt,name=o_string,json=oString"`
|
||||||
|
OString2 *string `protobuf:"bytes,name=ameno"`
|
||||||
|
OString3 *string `protobuf:"bytes,name=ameno"` // MATCH /duplicate tag name: 'ameno'/
|
||||||
|
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
||||||
|
XXX_unrecognized []byte `json:"-"`
|
||||||
|
XXX_sizecache int32 `json:"-"`
|
||||||
|
}
|
1
testdata/struct_tag.go
vendored
1
testdata/struct_tag.go
vendored
@ -25,6 +25,7 @@ type decodeAndValidateRequest struct {
|
|||||||
// No-reg test for bug https://github.com/mgechev/revive/issues/208
|
// No-reg test for bug https://github.com/mgechev/revive/issues/208
|
||||||
Tiret string `json:"-,"`
|
Tiret string `json:"-,"`
|
||||||
BadTiret string `json:"other,"` // MATCH /option can not be empty in JSON tag/
|
BadTiret string `json:"other,"` // MATCH /option can not be empty in JSON tag/
|
||||||
|
ForOmitzero string `json:"forOmitZero,omitzero"` // MATCH /unknown option 'omitzero' in JSON tag/
|
||||||
}
|
}
|
||||||
|
|
||||||
type RangeAllocation struct {
|
type RangeAllocation struct {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user