golangci-lint/pkg/golinters/enabled_linters.go

161 lines
4.5 KiB
Go

package golinters
import (
"context"
"fmt"
"sync"
"github.com/golangci/golangci-lint/pkg"
"github.com/golangci/golangci-lint/pkg/config"
"github.com/golangci/golangci-shared/pkg/analytics"
)
type LinterConfig struct {
EnabledByDefault bool
Desc string
Linter pkg.Linter
}
func enabledByDefault(linter pkg.Linter, desc string) LinterConfig {
return LinterConfig{
EnabledByDefault: true,
Linter: linter,
Desc: desc,
}
}
func disabledByDefault(linter pkg.Linter, desc string) LinterConfig {
return LinterConfig{
EnabledByDefault: false,
Linter: linter,
Desc: desc,
}
}
func GetAllSupportedLinterConfigs() []LinterConfig {
return []LinterConfig{
enabledByDefault(govet{}, "Vet examines Go source code and reports suspicious constructs, such as Printf calls whose arguments do not align with the format string"),
enabledByDefault(errcheck{}, "Errcheck is a program for checking for unchecked errors in go programs. These unchecked errors can be critical bugs in some cases"),
enabledByDefault(golint{}, "Golint differs from gofmt. Gofmt reformats Go source code, whereas golint prints out style mistakes."),
disabledByDefault(gofmt{}, "Gofmt checks whether code was gofmt-ed. By default this tool runs with -s option to check for code simplification."),
disabledByDefault(gofmt{useGoimports: true}, "Goimports does everything that gofmt does. Additionally it checks unused imports."),
}
}
func getAllSupportedLinters() []pkg.Linter {
var ret []pkg.Linter
for _, lc := range GetAllSupportedLinterConfigs() {
ret = append(ret, lc.Linter)
}
return ret
}
func getAllEnabledByDefaultLinters() []pkg.Linter {
var ret []pkg.Linter
for _, lc := range GetAllSupportedLinterConfigs() {
if lc.EnabledByDefault {
ret = append(ret, lc.Linter)
}
}
return ret
}
var supportedLintersByName map[string]pkg.Linter
var linterByNameMapOnce sync.Once
func getLinterByName(name string) pkg.Linter {
linterByNameMapOnce.Do(func() {
supportedLintersByName = make(map[string]pkg.Linter)
for _, lc := range GetAllSupportedLinterConfigs() {
supportedLintersByName[lc.Linter.Name()] = lc.Linter
}
})
return supportedLintersByName[name]
}
func lintersToMap(linters []pkg.Linter) map[string]pkg.Linter {
ret := map[string]pkg.Linter{}
for _, linter := range linters {
ret[linter.Name()] = linter
}
return ret
}
func validateEnabledDisabledLintersConfig(cfg *config.Run) error {
allNames := append([]string{}, cfg.EnabledLinters...)
allNames = append(allNames, cfg.DisabledLinters...)
for _, name := range allNames {
if getLinterByName(name) == nil {
return fmt.Errorf("no such linter %q", name)
}
}
if cfg.EnableAllLinters && cfg.DisableAllLinters {
return fmt.Errorf("--enable-all and --disable-all options must not be combined")
}
if cfg.DisableAllLinters {
if len(cfg.EnabledLinters) == 0 {
return fmt.Errorf("all linters were disabled, but no one linter was enabled: must enable at least one")
}
if len(cfg.DisabledLinters) != 0 {
return fmt.Errorf("can't combine options --disable-all and --disable %s", cfg.DisabledLinters[0])
}
}
if cfg.EnableAllLinters && len(cfg.EnabledLinters) != 0 {
return fmt.Errorf("can't combine options --enable-all and --enable %s", cfg.EnabledLinters[0])
}
enabledLintersSet := map[string]bool{}
for _, name := range cfg.EnabledLinters {
enabledLintersSet[name] = true
}
for _, name := range cfg.DisabledLinters {
if enabledLintersSet[name] {
return fmt.Errorf("linter %q can't be disabled and enabled at one moment", name)
}
}
return nil
}
func GetEnabledLinters(ctx context.Context, cfg *config.Run) ([]pkg.Linter, error) {
if err := validateEnabledDisabledLintersConfig(cfg); err != nil {
return nil, err
}
resultLintersSet := map[string]pkg.Linter{}
switch {
case cfg.EnableAllLinters:
resultLintersSet = lintersToMap(getAllSupportedLinters())
case cfg.DisableAllLinters:
break
default:
resultLintersSet = lintersToMap(getAllEnabledByDefaultLinters())
}
for _, name := range cfg.EnabledLinters {
resultLintersSet[name] = getLinterByName(name)
}
for _, name := range cfg.DisabledLinters {
delete(resultLintersSet, name)
}
var resultLinters []pkg.Linter
var resultLinterNames []string
for name, linter := range resultLintersSet {
resultLinters = append(resultLinters, linter)
resultLinterNames = append(resultLinterNames, name)
}
analytics.Log(ctx).Infof("Enabled linters: %s", resultLinterNames)
return resultLinters, nil
}