1
0
mirror of https://github.com/mgechev/revive.git synced 2025-10-08 22:41:54 +02:00
Files
revive/formatter/friendly.go
2025-07-12 07:32:39 -07:00

140 lines
3.5 KiB
Go

package formatter
import (
"bytes"
"cmp"
"fmt"
"io"
"slices"
"strings"
"text/tabwriter"
"github.com/fatih/color"
"github.com/mgechev/revive/lint"
)
// Friendly is an implementation of the Formatter interface
// which formats the errors to JSON.
type Friendly struct {
Metadata lint.FormatterMetadata
}
// Name returns the name of the formatter.
func (*Friendly) Name() string {
return "friendly"
}
// Format formats the failures gotten from the lint.
func (f *Friendly) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
var buf strings.Builder
errorMap := map[string]int{}
warningMap := map[string]int{}
totalErrors := 0
totalWarnings := 0
for failure := range failures {
sev := severity(config, failure)
f.printFriendlyFailure(&buf, failure, sev)
switch sev {
case lint.SeverityWarning:
warningMap[failure.RuleName]++
totalWarnings++
case lint.SeverityError:
errorMap[failure.RuleName]++
totalErrors++
}
}
f.printSummary(&buf, totalErrors, totalWarnings)
f.printStatistics(&buf, color.RedString("Errors:"), errorMap)
f.printStatistics(&buf, color.YellowString("Warnings:"), warningMap)
return buf.String(), nil
}
func (f *Friendly) printFriendlyFailure(sb *strings.Builder, failure lint.Failure, severity lint.Severity) {
f.printHeaderRow(sb, failure, severity)
f.printFilePosition(sb, failure)
sb.WriteString("\n\n")
}
var errorEmoji = color.RedString("✘")
var warningEmoji = color.YellowString("⚠")
func (*Friendly) printHeaderRow(sb *strings.Builder, failure lint.Failure, severity lint.Severity) {
emoji := warningEmoji
if severity == lint.SeverityError {
emoji = errorEmoji
}
sb.WriteString(table([][]string{{emoji, ruleDescriptionURL(failure.RuleName), color.GreenString(failure.Failure)}}))
}
func (*Friendly) printFilePosition(sb *strings.Builder, failure lint.Failure) {
fmt.Fprintf(sb, " %s:%d:%d", failure.Filename(), failure.Position.Start.Line, failure.Position.Start.Column)
}
type statEntry struct {
name string
failures int
}
func (*Friendly) printSummary(w io.Writer, errors, warnings int) {
emoji := warningEmoji
if errors > 0 {
emoji = errorEmoji
}
problemsLabel := "problems"
if errors+warnings == 1 {
problemsLabel = "problem"
}
warningsLabel := "warnings"
if warnings == 1 {
warningsLabel = "warning"
}
errorsLabel := "errors"
if errors == 1 {
errorsLabel = "error"
}
str := fmt.Sprintf("%d %s (%d %s, %d %s)", errors+warnings, problemsLabel, errors, errorsLabel, warnings, warningsLabel)
if errors > 0 {
fmt.Fprintf(w, "%s %s\n\n", emoji, color.RedString(str))
return
}
if warnings > 0 {
fmt.Fprintf(w, "%s %s\n\n", emoji, color.YellowString(str))
return
}
}
func (*Friendly) printStatistics(w io.Writer, header string, stats map[string]int) {
if len(stats) == 0 {
return
}
data := make([]statEntry, 0, len(stats))
for name, total := range stats {
data = append(data, statEntry{name, total})
}
slices.SortFunc(data, func(a, b statEntry) int {
return -cmp.Compare(a.failures, b.failures)
})
formatted := [][]string{}
for _, entry := range data {
formatted = append(formatted, []string{color.GreenString(fmt.Sprintf("%d", entry.failures)), entry.name})
}
fmt.Fprintln(w, header)
fmt.Fprintln(w, table(formatted))
}
func table(rows [][]string) string {
var buf bytes.Buffer
tw := tabwriter.NewWriter(&buf, 0, 0, 2, ' ', 0)
for _, row := range rows {
tw.Write([]byte{'\t'})
for _, col := range row {
tw.Write(append([]byte(col), '\t'))
}
tw.Write([]byte{'\n'})
}
tw.Flush()
return buf.String()
}