1
0
mirror of https://github.com/mgechev/revive.git synced 2025-01-10 03:17:11 +02:00
revive/rule/enforce-slice-style.go
Jameel Al-Aziz 17a0393326
enforce-slice-style: Support nil declaration enforcement (#974)
* enforce-slice-style: Support nil declaration enforcement

Add support for enforcing nil slice declarations as recommended by both
the Go Code Review and Uber style guides.

This initial version is quite strict in that it also prevents using
empty literal and make-style slices in struct assignments and as
function arguments.

* Add more tests

* docs
2024-02-26 11:58:31 +01:00

215 lines
4.8 KiB
Go

package rule
import (
"fmt"
"go/ast"
"sync"
"github.com/mgechev/revive/lint"
)
type enforceSliceStyleType string
const (
enforceSliceStyleTypeAny enforceSliceStyleType = "any"
enforceSliceStyleTypeMake enforceSliceStyleType = "make"
enforceSliceStyleTypeLiteral enforceSliceStyleType = "literal"
enforceSliceStyleTypeNil enforceSliceStyleType = "nil"
)
func sliceStyleFromString(s string) (enforceSliceStyleType, error) {
switch s {
case string(enforceSliceStyleTypeAny), "":
return enforceSliceStyleTypeAny, nil
case string(enforceSliceStyleTypeMake):
return enforceSliceStyleTypeMake, nil
case string(enforceSliceStyleTypeLiteral):
return enforceSliceStyleTypeLiteral, nil
case string(enforceSliceStyleTypeNil):
return enforceSliceStyleTypeNil, nil
default:
return enforceSliceStyleTypeAny, fmt.Errorf(
"invalid slice style: %s (expecting one of %v)",
s,
[]enforceSliceStyleType{
enforceSliceStyleTypeAny,
enforceSliceStyleTypeMake,
enforceSliceStyleTypeLiteral,
enforceSliceStyleTypeNil,
},
)
}
}
// EnforceSliceStyleRule implements a rule to enforce `make([]type)` over `[]type{}`.
type EnforceSliceStyleRule struct {
configured bool
enforceSliceStyle enforceSliceStyleType
sync.Mutex
}
func (r *EnforceSliceStyleRule) configure(arguments lint.Arguments) {
r.Lock()
defer r.Unlock()
if r.configured {
return
}
r.configured = true
if len(arguments) < 1 {
r.enforceSliceStyle = enforceSliceStyleTypeAny
return
}
enforceSliceStyle, ok := arguments[0].(string)
if !ok {
panic(fmt.Sprintf("Invalid argument '%v' for 'enforce-slice-style' rule. Expecting string, got %T", arguments[0], arguments[0]))
}
var err error
r.enforceSliceStyle, err = sliceStyleFromString(enforceSliceStyle)
if err != nil {
panic(fmt.Sprintf("Invalid argument to the enforce-slice-style rule: %v", err))
}
}
// Apply applies the rule to given file.
func (r *EnforceSliceStyleRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
r.configure(arguments)
if r.enforceSliceStyle == enforceSliceStyleTypeAny {
// this linter is not configured
return nil
}
var failures []lint.Failure
astFile := file.AST
ast.Inspect(astFile, func(n ast.Node) bool {
switch v := n.(type) {
case *ast.CompositeLit:
switch r.enforceSliceStyle {
case enforceSliceStyleTypeMake, enforceSliceStyleTypeNil:
// continue
default:
return true
}
if !r.isSliceType(v.Type) {
return true
}
if len(v.Elts) > 0 {
// not an empty slice
return true
}
var failureMessage string
if r.enforceSliceStyle == enforceSliceStyleTypeNil {
failureMessage = "use nil slice declaration (e.g. var args []type) instead of []type{}"
} else {
failureMessage = "use make([]type) instead of []type{} (or declare nil slice)"
}
failures = append(failures, lint.Failure{
Confidence: 1,
Node: v,
Category: "style",
Failure: failureMessage,
})
case *ast.CallExpr:
switch r.enforceSliceStyle {
case enforceSliceStyleTypeLiteral, enforceSliceStyleTypeNil:
default:
// skip any function calls, even if it's make([]type)
// we don't want to report it if literals are not enforced
return true
}
ident, ok := v.Fun.(*ast.Ident)
if !ok || ident.Name != "make" {
return true
}
if len(v.Args) < 2 {
// skip invalid make declarations
return true
}
if !r.isSliceType(v.Args[0]) {
// not a slice type
return true
}
arg, ok := v.Args[1].(*ast.BasicLit)
if !ok {
// skip invalid make declarations
return true
}
if arg.Value != "0" {
// skip slice with non-zero size
return true
}
if len(v.Args) > 2 {
arg, ok := v.Args[2].(*ast.BasicLit)
if !ok {
// skip invalid make declarations
return true
}
if arg.Value != "0" {
// skip non-zero capacity slice
return true
}
}
var failureMessage string
if r.enforceSliceStyle == enforceSliceStyleTypeNil {
failureMessage = "use nil slice declaration (e.g. var args []type) instead of make([]type, 0)"
} else {
failureMessage = "use []type{} instead of make([]type, 0) (or declare nil slice)"
}
failures = append(failures, lint.Failure{
Confidence: 1,
Node: v.Args[0],
Category: "style",
Failure: failureMessage,
})
}
return true
})
return failures
}
// Name returns the rule name.
func (*EnforceSliceStyleRule) Name() string {
return "enforce-slice-style"
}
func (r *EnforceSliceStyleRule) isSliceType(v ast.Expr) bool {
switch t := v.(type) {
case *ast.ArrayType:
if t.Len != nil {
// array
return false
}
// slice
return true
case *ast.Ident:
if t.Obj == nil {
return false
}
typeSpec, ok := t.Obj.Decl.(*ast.TypeSpec)
if !ok {
return false
}
return r.isSliceType(typeSpec.Type)
default:
return false
}
}