-
Notifications
You must be signed in to change notification settings - Fork 287
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Adds rule
max-control-nesting
(#967)
- Loading branch information
Showing
6 changed files
with
225 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,128 @@ | ||
package rule | ||
|
||
import ( | ||
"fmt" | ||
"go/ast" | ||
"sync" | ||
|
||
"github.com/mgechev/revive/lint" | ||
) | ||
|
||
// MaxControlNestingRule lints given else constructs. | ||
type MaxControlNestingRule struct { | ||
max int64 | ||
sync.Mutex | ||
} | ||
|
||
const defaultMaxControlNesting = 5 | ||
|
||
// Apply applies the rule to given file. | ||
func (r *MaxControlNestingRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure { | ||
r.configure(arguments) | ||
|
||
var failures []lint.Failure | ||
|
||
fileAst := file.AST | ||
|
||
walker := &lintMaxControlNesting{ | ||
onFailure: func(failure lint.Failure) { | ||
failures = append(failures, failure) | ||
}, | ||
max: int(r.max), | ||
} | ||
|
||
ast.Walk(walker, fileAst) | ||
|
||
return failures | ||
} | ||
|
||
// Name returns the rule name. | ||
func (*MaxControlNestingRule) Name() string { | ||
return "max-control-nesting" | ||
} | ||
|
||
type lintMaxControlNesting struct { | ||
max int | ||
onFailure func(lint.Failure) | ||
nestingLevelAcc int | ||
lastCtrlStmt ast.Node | ||
} | ||
|
||
func (w *lintMaxControlNesting) Visit(n ast.Node) ast.Visitor { | ||
if w.nestingLevelAcc > w.max { // we are visiting a node beyond the max nesting level | ||
w.onFailure(lint.Failure{ | ||
Failure: fmt.Sprintf("control flow nesting exceeds %d", w.max), | ||
Confidence: 1, | ||
Node: w.lastCtrlStmt, | ||
Category: "complexity", | ||
}) | ||
return nil // stop visiting deeper | ||
} | ||
|
||
switch v := n.(type) { | ||
case *ast.IfStmt: | ||
w.lastCtrlStmt = v | ||
w.walkControlledBlock(v.Body) // "then" branch block | ||
if v.Else != nil { | ||
w.walkControlledBlock(v.Else) // "else" branch block | ||
} | ||
return nil // stop re-visiting nesting blocks (already visited by w.walkControlledBlock) | ||
|
||
case *ast.ForStmt: | ||
w.lastCtrlStmt = v | ||
w.walkControlledBlock(v.Body) | ||
return nil // stop re-visiting nesting blocks (already visited by w.walkControlledBlock) | ||
|
||
case *ast.CaseClause: // switch case | ||
w.lastCtrlStmt = v | ||
for _, s := range v.Body { // visit each statement in the case clause | ||
w.walkControlledBlock(s) | ||
} | ||
return nil // stop re-visiting nesting blocks (already visited by w.walkControlledBlock) | ||
|
||
case *ast.CommClause: // select case | ||
w.lastCtrlStmt = v | ||
for _, s := range v.Body { // visit each statement in the select case clause | ||
w.walkControlledBlock(s) | ||
} | ||
return nil // stop re-visiting nesting blocks (already visited by w.walkControlledBlock) | ||
|
||
case *ast.FuncLit: | ||
walker := &lintMaxControlNesting{ | ||
onFailure: w.onFailure, | ||
max: w.max, | ||
} | ||
ast.Walk(walker, v.Body) | ||
return nil | ||
} | ||
|
||
return w | ||
} | ||
|
||
func (w *lintMaxControlNesting) walkControlledBlock(b ast.Node) { | ||
oldNestingLevel := w.nestingLevelAcc | ||
w.nestingLevelAcc++ | ||
ast.Walk(w, b) | ||
w.nestingLevelAcc = oldNestingLevel | ||
} | ||
|
||
func (r *MaxControlNestingRule) configure(arguments lint.Arguments) { | ||
r.Lock() | ||
defer r.Unlock() | ||
if !(r.max < 1) { | ||
return // max already set | ||
} | ||
|
||
if len(arguments) < 1 { | ||
r.max = defaultMaxControlNesting | ||
return | ||
} | ||
|
||
checkNumberOfArguments(1, arguments, r.Name()) | ||
|
||
max, ok := arguments[0].(int64) // Alt. non panicking version | ||
if !ok { | ||
panic(`invalid value passed as argument number to the "max-control-nesting" rule`) | ||
} | ||
r.max = max | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
package test | ||
|
||
import ( | ||
"testing" | ||
|
||
"github.com/mgechev/revive/lint" | ||
"github.com/mgechev/revive/rule" | ||
) | ||
|
||
func TestMaxControlNesting(t *testing.T) { | ||
testRule(t, "max-control-nesting", &rule.MaxControlNestingRule{}, &lint.RuleConfig{ | ||
Arguments: []any{int64(2)}}, | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
package fixtures | ||
|
||
func mcn() { | ||
if true { | ||
if true { | ||
if true { // MATCH /control flow nesting exceeds 2/ | ||
|
||
} | ||
} | ||
} else { | ||
if true { | ||
if true { // MATCH /control flow nesting exceeds 2/ | ||
if true { | ||
|
||
} | ||
} | ||
} | ||
} | ||
|
||
for { | ||
if true { | ||
for { // MATCH /control flow nesting exceeds 2/ | ||
} | ||
} | ||
} | ||
|
||
switch { | ||
case false: | ||
if true { | ||
|
||
} | ||
case true: | ||
if true { | ||
for { // MATCH /control flow nesting exceeds 2/ | ||
} | ||
} | ||
default: | ||
} | ||
|
||
select { | ||
case msg1 := <-c1: | ||
println("received", msg1) | ||
case msg2 := <-c2: | ||
if true { | ||
for { // MATCH /control flow nesting exceeds 2/ | ||
} | ||
} | ||
} | ||
|
||
if true { | ||
f1 := func() { | ||
if true { | ||
for { | ||
} | ||
} | ||
} | ||
} | ||
|
||
f1 := func() { | ||
for { | ||
if true { | ||
for { // MATCH /control flow nesting exceeds 2/ | ||
} | ||
} | ||
} | ||
} | ||
} |