1
0
mirror of https://github.com/mgechev/revive.git synced 2025-01-24 03:47:45 +02:00
revive/rule/package-comments.go

173 lines
4.4 KiB
Go
Raw Normal View History

2018-01-21 18:04:41 -08:00
package rule
2017-11-26 18:48:07 -08:00
import (
"fmt"
"go/ast"
"go/token"
"strings"
"sync"
2017-11-26 18:48:07 -08:00
2018-01-24 15:44:03 -08:00
"github.com/mgechev/revive/lint"
2017-11-26 18:48:07 -08:00
)
// PackageCommentsRule lints the package comments. It complains if
// there is no package comment, or if it is not of the right form.
// This has a notable false positive in that a package comment
// could rightfully appear in a different file of the same package,
// but that's not easy to fix since this linter is file-oriented.
type PackageCommentsRule struct {
checkPackageCommentCache sync.Map
}
2017-11-26 18:48:07 -08:00
// Apply applies the rule to given file.
func (r *PackageCommentsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
2018-01-24 15:44:03 -08:00
var failures []lint.Failure
2017-11-26 18:48:07 -08:00
if file.IsTest() {
2017-11-26 18:48:07 -08:00
return failures
}
2018-01-24 15:44:03 -08:00
onFailure := func(failure lint.Failure) {
2017-11-26 18:48:07 -08:00
failures = append(failures, failure)
}
2018-01-21 18:48:51 -08:00
fileAst := file.AST
w := &lintPackageComments{fileAst, file, onFailure, r}
2017-11-26 18:48:07 -08:00
ast.Walk(w, fileAst)
return failures
}
// Name returns the rule name.
2022-04-10 11:55:13 +02:00
func (*PackageCommentsRule) Name() string {
2017-11-26 18:48:07 -08:00
return "package-comments"
}
type lintPackageComments struct {
fileAst *ast.File
2018-01-24 15:44:03 -08:00
file *lint.File
onFailure func(lint.Failure)
rule *PackageCommentsRule
}
func (l *lintPackageComments) checkPackageComment() []lint.Failure {
// deduplicate warnings in package
if _, exists := l.rule.checkPackageCommentCache.LoadOrStore(l.file.Pkg, struct{}{}); exists {
return nil
}
var docFile *ast.File // which name is doc.go
var packageFile *ast.File // which name is $package.go
var firstFile *ast.File
var firstFileName string
var fileSource string
for name, file := range l.file.Pkg.Files() {
if file.AST.Doc != nil {
return nil
}
if name == "doc.go" {
docFile = file.AST
fileSource = "doc.go"
}
if name == file.AST.Name.String()+".go" {
packageFile = file.AST
}
if firstFileName == "" || firstFileName > name {
firstFile = file.AST
firstFileName = name
}
}
// prefer warning on doc.go, $package.go over first file
if docFile == nil {
docFile = packageFile
fileSource = l.fileAst.Name.String() + ".go"
}
if docFile == nil {
docFile = firstFile
fileSource = firstFileName
}
if docFile != nil {
pkgFile := l.file.Pkg.Files()[fileSource]
return []lint.Failure{{
Category: "comments",
Position: lint.FailurePosition{
Start: pkgFile.ToPosition(docFile.Pos()),
End: pkgFile.ToPosition(docFile.Name.End()),
},
Confidence: 1,
Failure: "should have a package comment",
}}
}
return nil
2017-11-26 18:48:07 -08:00
}
func (l *lintPackageComments) Visit(_ ast.Node) ast.Visitor {
2018-01-23 19:13:02 -08:00
if l.file.IsTest() {
return nil
}
2017-11-26 18:48:07 -08:00
prefix := "Package " + l.fileAst.Name.Name + " "
// Look for a detached package comment.
// First, scan for the last comment that occurs before the "package" keyword.
var lastCG *ast.CommentGroup
for _, cg := range l.fileAst.Comments {
if cg.Pos() > l.fileAst.Package {
// Gone past "package" keyword.
break
}
lastCG = cg
}
if lastCG != nil && strings.HasPrefix(lastCG.Text(), prefix) {
endPos := l.file.ToPosition(lastCG.End())
pkgPos := l.file.ToPosition(l.fileAst.Package)
if endPos.Line+1 < pkgPos.Line {
// There isn't a great place to anchor this error;
// the start of the blank lines between the doc and the package statement
// is at least pointing at the location of the problem.
pos := token.Position{
Filename: endPos.Filename,
// Offset not set; it is non-trivial, and doesn't appear to be needed.
Line: endPos.Line + 1,
Column: 1,
}
2018-01-24 15:44:03 -08:00
l.onFailure(lint.Failure{
2018-01-23 19:13:02 -08:00
Category: "comments",
2018-01-24 15:44:03 -08:00
Position: lint.FailurePosition{
2018-01-23 19:13:02 -08:00
Start: pos,
End: pos,
},
2017-11-26 18:48:07 -08:00
Confidence: 0.9,
2018-01-23 19:13:02 -08:00
Failure: "package comment is detached; there should be no blank lines between it and the package statement",
2017-11-26 18:48:07 -08:00
})
return nil
}
}
if l.fileAst.Doc == nil {
for _, failure := range l.checkPackageComment() {
l.onFailure(failure)
}
2017-11-26 18:48:07 -08:00
return nil
}
s := l.fileAst.Doc.Text()
if ts := strings.TrimLeft(s, " \t"); ts != s {
2018-01-24 15:44:03 -08:00
l.onFailure(lint.Failure{
2018-01-23 19:13:02 -08:00
Category: "comments",
2017-11-26 18:48:07 -08:00
Node: l.fileAst.Doc,
2018-01-23 19:13:02 -08:00
Confidence: 1,
Failure: "package comment should not have leading space",
2017-11-26 18:48:07 -08:00
})
s = ts
}
// Only non-main packages need to keep to this form.
2018-01-23 19:13:02 -08:00
if !l.file.Pkg.IsMain() && !strings.HasPrefix(s, prefix) {
2018-01-24 15:44:03 -08:00
l.onFailure(lint.Failure{
2018-01-23 19:13:02 -08:00
Category: "comments",
2017-11-26 18:48:07 -08:00
Node: l.fileAst.Doc,
2018-01-23 19:13:02 -08:00
Confidence: 1,
Failure: fmt.Sprintf(`package comment should be of the form "%s..."`, prefix),
2017-11-26 18:48:07 -08:00
})
}
return nil
}