
Use build.Import instead of manual parser.ParseFile and paths traversal. It allows: 1. support build tags for all linters. 2. analyze files only for current GOOS/GOARCH: less false-positives. 3. analyze xtest packages (*_test) by golint: upstream golint and gometalinter can't do it! And don't break analysis on the first xtest package like it was before. 4. proper handling of xtest packages for linters like goconst where package boundary is important: less false-positives is expected. Also: 1. reuse AST parsing for golint and goconst: minor speedup. 2. allow to specify path (not only name) regexp for --skip-files and --skip-dirs 3. add more default exclude filters for golint about commits: `(comment on exported (method|function)|should have( a package)? comment|comment should be of the form)` 4. print skipped dir in verbose (-v) mode 5. refactor per-linter tests: declare arguments in comments, run only one linter and in combination with slow linter
133 lines
3.1 KiB
Go
133 lines
3.1 KiB
Go
package golinters
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"go/token"
|
|
|
|
gofmtAPI "github.com/golangci/gofmt/gofmt"
|
|
goimportsAPI "github.com/golangci/gofmt/goimports"
|
|
"github.com/golangci/golangci-lint/pkg/lint/linter"
|
|
"github.com/golangci/golangci-lint/pkg/result"
|
|
"github.com/sirupsen/logrus"
|
|
"sourcegraph.com/sourcegraph/go-diff/diff"
|
|
)
|
|
|
|
type Gofmt struct {
|
|
UseGoimports bool
|
|
}
|
|
|
|
func (g Gofmt) Name() string {
|
|
if g.UseGoimports {
|
|
return "goimports"
|
|
}
|
|
|
|
return "gofmt"
|
|
}
|
|
|
|
func (g Gofmt) Desc() string {
|
|
if g.UseGoimports {
|
|
return "Goimports does everything that gofmt does. Additionally it checks unused imports"
|
|
}
|
|
|
|
return "Gofmt checks whether code was gofmt-ed. By default this tool runs with -s option to check for code simplification"
|
|
}
|
|
|
|
func getFirstDeletedAndAddedLineNumberInHunk(h *diff.Hunk) (int, int, error) {
|
|
lines := bytes.Split(h.Body, []byte{'\n'})
|
|
lineNumber := int(h.OrigStartLine - 1)
|
|
firstAddedLineNumber := -1
|
|
for _, line := range lines {
|
|
lineNumber++
|
|
|
|
if len(line) == 0 {
|
|
continue
|
|
}
|
|
if line[0] == '+' && firstAddedLineNumber == -1 {
|
|
firstAddedLineNumber = lineNumber
|
|
}
|
|
if line[0] == '-' {
|
|
return lineNumber, firstAddedLineNumber, nil
|
|
}
|
|
}
|
|
|
|
return 0, firstAddedLineNumber, fmt.Errorf("didn't find deletion line in hunk %s", string(h.Body))
|
|
}
|
|
|
|
func (g Gofmt) extractIssuesFromPatch(patch string) ([]result.Issue, error) {
|
|
diffs, err := diff.ParseMultiFileDiff([]byte(patch))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't parse patch: %s", err)
|
|
}
|
|
|
|
if len(diffs) == 0 {
|
|
return nil, fmt.Errorf("got no diffs from patch parser: %v", diffs)
|
|
}
|
|
|
|
issues := []result.Issue{}
|
|
for _, d := range diffs {
|
|
if len(d.Hunks) == 0 {
|
|
logrus.Warnf("Got no hunks in diff %+v", d)
|
|
continue
|
|
}
|
|
|
|
for _, hunk := range d.Hunks {
|
|
deletedLine, addedLine, err := getFirstDeletedAndAddedLineNumberInHunk(hunk)
|
|
if err != nil {
|
|
logrus.Infof("Can't get first deleted line number for hunk: %s", err)
|
|
if addedLine > 1 {
|
|
deletedLine = addedLine - 1 // use previous line, TODO: use both prev and next lines
|
|
} else {
|
|
deletedLine = 1
|
|
}
|
|
}
|
|
|
|
text := "File is not gofmt-ed with -s"
|
|
if g.UseGoimports {
|
|
text = "File is not goimports-ed"
|
|
}
|
|
i := result.Issue{
|
|
FromLinter: g.Name(),
|
|
Pos: token.Position{
|
|
Filename: d.NewName,
|
|
Line: deletedLine,
|
|
},
|
|
Text: text,
|
|
}
|
|
issues = append(issues, i)
|
|
}
|
|
}
|
|
|
|
return issues, nil
|
|
}
|
|
|
|
func (g Gofmt) Run(ctx context.Context, lintCtx *linter.Context) ([]result.Issue, error) {
|
|
var issues []result.Issue
|
|
|
|
for _, f := range lintCtx.PkgProgram.Files(lintCtx.Cfg.Run.AnalyzeTests) {
|
|
var diff []byte
|
|
var err error
|
|
if g.UseGoimports {
|
|
diff, err = goimportsAPI.Run(f)
|
|
} else {
|
|
diff, err = gofmtAPI.Run(f, lintCtx.Settings().Gofmt.Simplify)
|
|
}
|
|
if err != nil { // TODO: skip
|
|
return nil, err
|
|
}
|
|
if diff == nil {
|
|
continue
|
|
}
|
|
|
|
is, err := g.extractIssuesFromPatch(string(diff))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't extract issues from gofmt diff output %q: %s", string(diff), err)
|
|
}
|
|
|
|
issues = append(issues, is...)
|
|
}
|
|
|
|
return issues, nil
|
|
}
|