
1. Log all warnings, don't hide none of them 2. Write fatal messages (stop analysis) with error log level 3. Remove ugly timestamp counter from logrus output 4. Print nested module prefix in log 5. Make logger abstraction: no global logging anymore 6. Refactor config reading to config.FileReader struct to avoid passing logger into every function 7. Replace exit codes hardcoding with constants in exitcodes package 8. Fail test if any warning was logged 9. Fix calculation of relative path if we analyze parent dir ../ 10. Move Runner initialization from Executor to NewRunner func 11. Log every AST parsing error 12. Properly print used config file path in verbose mode 13. Print package files if only 1 package is analyzedin verbose mode, print not compiling packages in verbose mode 14. Forbid usage of github.com/sirupsen/logrus by DepGuard linter 15. Add default ignore pattern to folint: "comment on exported const"
277 lines
6.6 KiB
Go
277 lines
6.6 KiB
Go
package lint
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"runtime/debug"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/golangci/golangci-lint/pkg/config"
|
|
"github.com/golangci/golangci-lint/pkg/lint/astcache"
|
|
"github.com/golangci/golangci-lint/pkg/lint/linter"
|
|
"github.com/golangci/golangci-lint/pkg/logutils"
|
|
"github.com/golangci/golangci-lint/pkg/result"
|
|
"github.com/golangci/golangci-lint/pkg/result/processors"
|
|
"github.com/golangci/golangci-lint/pkg/timeutils"
|
|
)
|
|
|
|
type Runner struct {
|
|
Processors []processors.Processor
|
|
Log logutils.Log
|
|
}
|
|
|
|
func NewRunner(astCache *astcache.Cache, cfg *config.Config, log logutils.Log) (*Runner, error) {
|
|
icfg := cfg.Issues
|
|
excludePatterns := icfg.ExcludePatterns
|
|
if icfg.UseDefaultExcludes {
|
|
excludePatterns = append(excludePatterns, config.GetDefaultExcludePatternsStrings()...)
|
|
}
|
|
|
|
var excludeTotalPattern string
|
|
if len(excludePatterns) != 0 {
|
|
excludeTotalPattern = fmt.Sprintf("(%s)", strings.Join(excludePatterns, "|"))
|
|
}
|
|
|
|
skipFilesProcessor, err := processors.NewSkipFiles(cfg.Run.SkipFiles)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Runner{
|
|
Processors: []processors.Processor{
|
|
processors.NewPathPrettifier(), // must be before diff, nolint and exclude autogenerated processor at least
|
|
processors.NewCgo(),
|
|
skipFilesProcessor,
|
|
|
|
processors.NewAutogeneratedExclude(astCache),
|
|
processors.NewExclude(excludeTotalPattern),
|
|
processors.NewNolint(astCache),
|
|
|
|
processors.NewUniqByLine(),
|
|
processors.NewDiff(icfg.Diff, icfg.DiffFromRevision, icfg.DiffPatchFilePath),
|
|
processors.NewMaxPerFileFromLinter(),
|
|
processors.NewMaxSameIssues(icfg.MaxSameIssues, log.Child("max_same_issues")),
|
|
processors.NewMaxFromLinter(icfg.MaxIssuesPerLinter, log.Child("max_from_linter")),
|
|
},
|
|
Log: log,
|
|
}, nil
|
|
}
|
|
|
|
type lintRes struct {
|
|
linter linter.Config
|
|
err error
|
|
issues []result.Issue
|
|
}
|
|
|
|
func (r Runner) runLinterSafe(ctx context.Context, lintCtx *linter.Context, lc linter.Config) (ret []result.Issue, err error) {
|
|
defer func() {
|
|
if panicData := recover(); panicData != nil {
|
|
err = fmt.Errorf("panic occured: %s", panicData)
|
|
r.Log.Warnf("Panic stack trace: %s", debug.Stack())
|
|
}
|
|
}()
|
|
|
|
specificLintCtx := *lintCtx
|
|
specificLintCtx.Log = lintCtx.Log.Child(lc.Linter.Name())
|
|
issues, err := lc.Linter.Run(ctx, &specificLintCtx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, i := range issues {
|
|
i.FromLinter = lc.Linter.Name()
|
|
}
|
|
|
|
return issues, nil
|
|
}
|
|
|
|
func (r Runner) runWorker(ctx context.Context, lintCtx *linter.Context, tasksCh <-chan linter.Config, lintResultsCh chan<- lintRes, name string) {
|
|
sw := timeutils.NewStopwatch(name, r.Log)
|
|
defer sw.Print()
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case lc, ok := <-tasksCh:
|
|
if !ok {
|
|
return
|
|
}
|
|
if ctx.Err() != nil {
|
|
// XXX: if check it in only int a select
|
|
// it's possible to not enter to this case until tasksCh is empty.
|
|
return
|
|
}
|
|
var issues []result.Issue
|
|
var err error
|
|
sw.TrackStage(lc.Linter.Name(), func() {
|
|
issues, err = r.runLinterSafe(ctx, lintCtx, lc)
|
|
})
|
|
lintResultsCh <- lintRes{
|
|
linter: lc,
|
|
err: err,
|
|
issues: issues,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r Runner) logWorkersStat(workersFinishTimes []time.Time) {
|
|
lastFinishTime := workersFinishTimes[0]
|
|
for _, t := range workersFinishTimes {
|
|
if t.After(lastFinishTime) {
|
|
lastFinishTime = t
|
|
}
|
|
}
|
|
|
|
logStrings := []string{}
|
|
for i, t := range workersFinishTimes {
|
|
if t.Equal(lastFinishTime) {
|
|
continue
|
|
}
|
|
|
|
logStrings = append(logStrings, fmt.Sprintf("#%d: %s", i+1, lastFinishTime.Sub(t)))
|
|
}
|
|
|
|
r.Log.Infof("Workers idle times: %s", strings.Join(logStrings, ", "))
|
|
}
|
|
|
|
func getSortedLintersConfigs(linters []linter.Config) []linter.Config {
|
|
ret := make([]linter.Config, len(linters))
|
|
copy(ret, linters)
|
|
|
|
sort.Slice(ret, func(i, j int) bool {
|
|
return ret[i].GetSpeed() < ret[j].GetSpeed()
|
|
})
|
|
|
|
return ret
|
|
}
|
|
|
|
func (r *Runner) runWorkers(ctx context.Context, lintCtx *linter.Context, linters []linter.Config) <-chan lintRes {
|
|
tasksCh := make(chan linter.Config, len(linters))
|
|
lintResultsCh := make(chan lintRes, len(linters))
|
|
var wg sync.WaitGroup
|
|
|
|
workersFinishTimes := make([]time.Time, lintCtx.Cfg.Run.Concurrency)
|
|
|
|
for i := 0; i < lintCtx.Cfg.Run.Concurrency; i++ {
|
|
wg.Add(1)
|
|
go func(i int) {
|
|
defer wg.Done()
|
|
name := fmt.Sprintf("worker.%d", i+1)
|
|
r.runWorker(ctx, lintCtx, tasksCh, lintResultsCh, name)
|
|
workersFinishTimes[i] = time.Now()
|
|
}(i)
|
|
}
|
|
|
|
lcs := getSortedLintersConfigs(linters)
|
|
for _, lc := range lcs {
|
|
tasksCh <- lc
|
|
}
|
|
close(tasksCh)
|
|
|
|
go func() {
|
|
wg.Wait()
|
|
close(lintResultsCh)
|
|
|
|
r.logWorkersStat(workersFinishTimes)
|
|
}()
|
|
|
|
return lintResultsCh
|
|
}
|
|
|
|
func (r Runner) processLintResults(ctx context.Context, inCh <-chan lintRes) <-chan lintRes {
|
|
outCh := make(chan lintRes, 64)
|
|
|
|
go func() {
|
|
sw := timeutils.NewStopwatch("processing", r.Log)
|
|
|
|
defer close(outCh)
|
|
|
|
for res := range inCh {
|
|
if res.err != nil {
|
|
r.Log.Warnf("Can't run linter %s: %s", res.linter.Linter.Name(), res.err)
|
|
continue
|
|
}
|
|
|
|
if len(res.issues) != 0 {
|
|
res.issues = r.processIssues(ctx, res.issues, sw)
|
|
outCh <- res
|
|
}
|
|
}
|
|
|
|
// finalize processors: logging, clearing, no heavy work here
|
|
|
|
for _, p := range r.Processors {
|
|
sw.TrackStage(p.Name(), func() {
|
|
p.Finish()
|
|
})
|
|
}
|
|
|
|
sw.PrintStages()
|
|
}()
|
|
|
|
return outCh
|
|
}
|
|
|
|
func collectIssues(ctx context.Context, resCh <-chan lintRes) <-chan result.Issue {
|
|
retIssues := make(chan result.Issue, 1024)
|
|
go func() {
|
|
defer close(retIssues)
|
|
|
|
for res := range resCh {
|
|
if len(res.issues) == 0 {
|
|
continue
|
|
}
|
|
|
|
for _, i := range res.issues {
|
|
retIssues <- i
|
|
}
|
|
}
|
|
}()
|
|
|
|
return retIssues
|
|
}
|
|
|
|
func (r Runner) Run(ctx context.Context, linters []linter.Config, lintCtx *linter.Context) <-chan result.Issue {
|
|
lintResultsCh := r.runWorkers(ctx, lintCtx, linters)
|
|
processedLintResultsCh := r.processLintResults(ctx, lintResultsCh)
|
|
if ctx.Err() != nil {
|
|
// XXX: always process issues, even if timeout occured
|
|
finishedLintersN := 0
|
|
for range processedLintResultsCh {
|
|
finishedLintersN++
|
|
}
|
|
|
|
r.Log.Errorf("%d/%d linters finished: deadline exceeded: try increase it by passing --deadline option",
|
|
finishedLintersN, len(linters))
|
|
}
|
|
|
|
return collectIssues(ctx, processedLintResultsCh)
|
|
}
|
|
|
|
func (r *Runner) processIssues(ctx context.Context, issues []result.Issue, sw *timeutils.Stopwatch) []result.Issue {
|
|
for _, p := range r.Processors {
|
|
var newIssues []result.Issue
|
|
var err error
|
|
sw.TrackStage(p.Name(), func() {
|
|
newIssues, err = p.Process(issues)
|
|
})
|
|
|
|
if err != nil {
|
|
r.Log.Warnf("Can't process result by %s processor: %s", p.Name(), err)
|
|
} else {
|
|
issues = newIssues
|
|
}
|
|
|
|
if issues == nil {
|
|
issues = []result.Issue{}
|
|
}
|
|
}
|
|
|
|
return issues
|
|
}
|