1
0
mirror of https://github.com/mgechev/revive.git synced 2024-11-24 08:32:22 +02:00
revive/config/config.go

234 lines
5.3 KiB
Go
Raw Normal View History

2020-10-29 15:11:52 +02:00
package config
2018-01-28 02:22:17 +02:00
import (
2020-10-29 15:11:52 +02:00
"errors"
2018-01-28 02:22:17 +02:00
"fmt"
"os"
2018-02-04 04:56:45 +02:00
2018-01-28 02:22:17 +02:00
"github.com/mgechev/revive/formatter"
2020-10-29 15:11:52 +02:00
"github.com/BurntSushi/toml"
2018-01-28 02:22:17 +02:00
"github.com/mgechev/revive/lint"
"github.com/mgechev/revive/rule"
)
var defaultRules = []lint.Rule{
&rule.VarDeclarationsRule{},
&rule.PackageCommentsRule{},
&rule.DotImportsRule{},
&rule.BlankImportsRule{},
&rule.ExportedRule{},
2018-05-27 06:28:31 +02:00
&rule.VarNamingRule{},
2018-05-27 01:14:36 +02:00
&rule.IndentErrorFlowRule{},
2018-01-28 02:22:17 +02:00
&rule.RangeRule{},
&rule.ErrorfRule{},
2018-05-27 06:28:31 +02:00
&rule.ErrorNamingRule{},
2018-01-28 02:22:17 +02:00
&rule.ErrorStringsRule{},
2018-05-27 06:28:31 +02:00
&rule.ReceiverNamingRule{},
2018-01-28 02:22:17 +02:00
&rule.IncrementDecrementRule{},
&rule.ErrorReturnRule{},
&rule.UnexportedReturnRule{},
2018-05-27 06:28:31 +02:00
&rule.TimeNamingRule{},
&rule.ContextKeysType{},
&rule.ContextAsArgumentRule{},
&rule.IfReturnRule{},
&rule.EmptyBlockRule{},
&rule.SuperfluousElseRule{},
&rule.UnusedParamRule{},
&rule.UnreachableCodeRule{},
&rule.RedefinesBuiltinIDRule{},
2018-01-28 02:22:17 +02:00
}
var allRules = append([]lint.Rule{
&rule.ArgumentsLimitRule{},
&rule.CyclomaticRule{},
2018-05-27 06:28:31 +02:00
&rule.FileHeaderRule{},
&rule.ConfusingNamingRule{},
&rule.GetReturnRule{},
&rule.ModifiesParamRule{},
&rule.ConfusingResultsRule{},
&rule.DeepExitRule{},
2018-07-17 21:21:27 +02:00
&rule.AddConstantRule{},
&rule.FlagParamRule{},
&rule.UnnecessaryStmtRule{},
2018-07-28 18:07:31 +02:00
&rule.StructTagRule{},
&rule.ModifiesValRecRule{},
&rule.ConstantLogicalExprRule{},
&rule.BoolLiteralRule{},
&rule.ImportsBlacklistRule{},
&rule.FunctionResultsLimitRule{},
&rule.MaxPublicStructsRule{},
&rule.RangeValInClosureRule{},
&rule.RangeValAddress{},
&rule.WaitGroupByValueRule{},
&rule.AtomicRule{},
&rule.EmptyLinesRule{},
&rule.LineLengthLimitRule{},
2018-10-31 16:32:23 +02:00
&rule.CallToGCRule{},
&rule.DuplicatedImportsRule{},
&rule.ImportShadowingRule{},
&rule.BareReturnRule{},
&rule.UnusedReceiverRule{},
&rule.UnhandledErrorRule{},
2019-12-14 09:27:06 +02:00
&rule.CognitiveComplexityRule{},
&rule.StringOfIntRule{},
&rule.StringFormatRule{},
&rule.EarlyReturnRule{},
2020-05-09 17:10:34 +02:00
&rule.UnconditionalRecursionRule{},
&rule.IdenticalBranchesRule{},
2020-05-24 20:49:49 +02:00
&rule.DeferRule{},
&rule.UnexportedNamingRule{},
2021-03-21 00:45:30 +02:00
&rule.FunctionLength{},
&rule.NestedStructs{},
2021-08-16 00:30:08 +02:00
&rule.UselessBreak{},
2021-10-03 15:35:13 +02:00
&rule.TimeEqualRule{},
&rule.BannedCharsRule{},
&rule.OptimizeOperandsOrderRule{},
2022-03-29 20:25:38 +02:00
&rule.UseAnyRule{},
2022-04-18 18:45:42 +02:00
&rule.DataRaceRule{},
&rule.CommentSpacingsRule{},
2018-01-28 02:22:17 +02:00
}, defaultRules...)
var allFormatters = []lint.Formatter{
2018-05-26 21:08:02 +02:00
&formatter.Stylish{},
2018-05-26 22:47:13 +02:00
&formatter.Friendly{},
2018-01-28 03:01:18 +02:00
&formatter.JSON{},
&formatter.NDJSON{},
2018-01-28 03:01:18 +02:00
&formatter.Default{},
2018-09-17 20:57:56 +02:00
&formatter.Unix{},
&formatter.Checkstyle{},
&formatter.Plain{},
2021-04-05 20:54:33 +02:00
&formatter.Sarif{},
2018-01-28 02:22:17 +02:00
}
func getFormatters() map[string]lint.Formatter {
result := map[string]lint.Formatter{}
for _, f := range allFormatters {
result[f.Name()] = f
}
return result
}
// GetLintingRules yields the linting rules that must be applied by the linter
func GetLintingRules(config *lint.Config, extraRules []lint.Rule) ([]lint.Rule, error) {
2018-01-28 02:22:17 +02:00
rulesMap := map[string]lint.Rule{}
for _, r := range allRules {
rulesMap[r.Name()] = r
}
for _, r := range extraRules {
if _, ok := rulesMap[r.Name()]; ok {
continue
}
rulesMap[r.Name()] = r
}
2018-01-28 02:22:17 +02:00
var lintingRules []lint.Rule
for name, ruleConfig := range config.Rules {
2022-04-10 11:55:13 +02:00
r, ok := rulesMap[name]
2018-01-28 02:22:17 +02:00
if !ok {
2020-10-29 15:11:52 +02:00
return nil, fmt.Errorf("cannot find rule: %s", name)
2018-01-28 02:22:17 +02:00
}
if ruleConfig.Disabled {
continue // skip disabled rules
}
2022-04-10 11:55:13 +02:00
lintingRules = append(lintingRules, r)
2018-01-28 02:22:17 +02:00
}
2020-10-29 15:11:52 +02:00
return lintingRules, nil
2018-01-28 02:22:17 +02:00
}
func parseConfig(path string, config *lint.Config) error {
file, err := os.ReadFile(path)
2018-01-28 02:22:17 +02:00
if err != nil {
return errors.New("cannot read the config file")
2018-01-28 02:22:17 +02:00
}
2020-10-29 15:11:52 +02:00
_, err = toml.Decode(string(file), config)
2018-01-28 02:22:17 +02:00
if err != nil {
return fmt.Errorf("cannot parse the config file: %v", err)
2018-01-28 02:22:17 +02:00
}
return nil
2018-01-28 02:22:17 +02:00
}
func normalizeConfig(config *lint.Config) {
2021-09-30 22:37:44 +02:00
if len(config.Rules) == 0 {
config.Rules = map[string]lint.RuleConfig{}
2018-01-28 03:01:18 +02:00
}
2021-09-30 22:37:44 +02:00
if config.EnableAllRules {
// Add to the configuration all rules not yet present in it
2022-04-10 11:55:13 +02:00
for _, r := range allRules {
ruleName := r.Name()
2021-09-30 22:37:44 +02:00
_, alreadyInConf := config.Rules[ruleName]
if alreadyInConf {
continue
}
// Add the rule with an empty conf for
config.Rules[ruleName] = lint.RuleConfig{}
}
}
2018-01-28 02:22:17 +02:00
severity := config.Severity
if severity != "" {
for k, v := range config.Rules {
if v.Severity == "" {
v.Severity = severity
}
config.Rules[k] = v
}
for k, v := range config.Directives {
if v.Severity == "" {
v.Severity = severity
}
config.Directives[k] = v
}
2018-01-28 02:22:17 +02:00
}
}
const defaultConfidence = 0.8
2020-10-29 15:11:52 +02:00
// GetConfig yields the configuration
func GetConfig(configPath string) (*lint.Config, error) {
config := &lint.Config{}
switch {
case configPath != "":
config.Confidence = defaultConfidence
err := parseConfig(configPath, config)
2020-10-29 15:11:52 +02:00
if err != nil {
return nil, err
}
default: // no configuration provided
config = defaultConfig()
2018-01-28 02:22:17 +02:00
}
2018-01-28 02:22:17 +02:00
normalizeConfig(config)
2020-10-29 15:11:52 +02:00
return config, nil
2018-01-28 02:22:17 +02:00
}
2020-10-29 15:11:52 +02:00
// GetFormatter yields the formatter for lint failures
func GetFormatter(formatterName string) (lint.Formatter, error) {
2018-01-28 02:22:17 +02:00
formatters := getFormatters()
2022-04-10 11:55:13 +02:00
fmtr := formatters["default"]
2018-01-28 02:22:17 +02:00
if formatterName != "" {
f, ok := formatters[formatterName]
if !ok {
2020-10-29 15:11:52 +02:00
return nil, fmt.Errorf("unknown formatter %v", formatterName)
2018-01-28 02:22:17 +02:00
}
2022-04-10 11:55:13 +02:00
fmtr = f
2018-01-28 02:22:17 +02:00
}
2022-04-10 11:55:13 +02:00
return fmtr, nil
}
2018-01-28 02:22:17 +02:00
func defaultConfig() *lint.Config {
defaultConfig := lint.Config{
Confidence: defaultConfidence,
2018-01-28 02:22:17 +02:00
Severity: lint.SeverityWarning,
Rules: map[string]lint.RuleConfig{},
}
for _, r := range defaultRules {
defaultConfig.Rules[r.Name()] = lint.RuleConfig{}
}
return &defaultConfig
}