mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2024-12-12 11:36:29 +00:00
113 lines
3.2 KiB
Go
113 lines
3.2 KiB
Go
|
package golinters
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
|
||
|
"github.com/OpenPeeDeeP/depguard"
|
||
|
"golang.org/x/tools/go/analysis"
|
||
|
"golang.org/x/tools/go/loader" //nolint:staticcheck // require changes in github.com/OpenPeeDeeP/depguard
|
||
|
|
||
|
"github.com/golangci/golangci-lint/pkg/golinters/goanalysis"
|
||
|
"github.com/golangci/golangci-lint/pkg/lint/linter"
|
||
|
"github.com/golangci/golangci-lint/pkg/result"
|
||
|
)
|
||
|
|
||
|
func setDepguardListType(dg *depguard.Depguard, lintCtx *linter.Context) error {
|
||
|
listType := lintCtx.Settings().Depguard.ListType
|
||
|
var found bool
|
||
|
dg.ListType, found = depguard.StringToListType[strings.ToLower(listType)]
|
||
|
if !found {
|
||
|
if listType != "" {
|
||
|
return fmt.Errorf("unsure what list type %s is", listType)
|
||
|
}
|
||
|
dg.ListType = depguard.LTBlacklist
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func setupDepguardPackages(dg *depguard.Depguard, lintCtx *linter.Context) {
|
||
|
if dg.ListType == depguard.LTBlacklist {
|
||
|
// if the list type was a blacklist the packages with error messages should
|
||
|
// be included in the blacklist package list
|
||
|
|
||
|
noMessagePackages := make(map[string]bool)
|
||
|
for _, pkg := range dg.Packages {
|
||
|
noMessagePackages[pkg] = true
|
||
|
}
|
||
|
|
||
|
for pkg := range lintCtx.Settings().Depguard.PackagesWithErrorMessage {
|
||
|
if _, ok := noMessagePackages[pkg]; !ok {
|
||
|
dg.Packages = append(dg.Packages, pkg)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func NewDepguard() *goanalysis.Linter {
|
||
|
const linterName = "depguard"
|
||
|
var mu sync.Mutex
|
||
|
var resIssues []goanalysis.Issue
|
||
|
|
||
|
analyzer := &analysis.Analyzer{
|
||
|
Name: linterName,
|
||
|
Doc: goanalysis.TheOnlyanalyzerDoc,
|
||
|
}
|
||
|
return goanalysis.NewLinter(
|
||
|
linterName,
|
||
|
"Go linter that checks if package imports are in a list of acceptable packages",
|
||
|
[]*analysis.Analyzer{analyzer},
|
||
|
nil,
|
||
|
).WithContextSetter(func(lintCtx *linter.Context) {
|
||
|
dgSettings := &lintCtx.Settings().Depguard
|
||
|
analyzer.Run = func(pass *analysis.Pass) (interface{}, error) {
|
||
|
prog := goanalysis.MakeFakeLoaderProgram(pass)
|
||
|
dg := &depguard.Depguard{
|
||
|
Packages: dgSettings.Packages,
|
||
|
IncludeGoRoot: dgSettings.IncludeGoRoot,
|
||
|
}
|
||
|
if err := setDepguardListType(dg, lintCtx); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
setupDepguardPackages(dg, lintCtx)
|
||
|
|
||
|
loadConfig := &loader.Config{
|
||
|
Cwd: "", // fallbacked to os.Getcwd
|
||
|
Build: nil, // fallbacked to build.Default
|
||
|
}
|
||
|
issues, err := dg.Run(loadConfig, prog)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if len(issues) == 0 {
|
||
|
return nil, nil
|
||
|
}
|
||
|
msgSuffix := "is in the blacklist"
|
||
|
if dg.ListType == depguard.LTWhitelist {
|
||
|
msgSuffix = "is not in the whitelist"
|
||
|
}
|
||
|
res := make([]goanalysis.Issue, 0, len(issues))
|
||
|
for _, i := range issues {
|
||
|
userSuppliedMsgSuffix := dgSettings.PackagesWithErrorMessage[i.PackageName]
|
||
|
if userSuppliedMsgSuffix != "" {
|
||
|
userSuppliedMsgSuffix = ": " + userSuppliedMsgSuffix
|
||
|
}
|
||
|
res = append(res, goanalysis.NewIssue(&result.Issue{
|
||
|
Pos: i.Position,
|
||
|
Text: fmt.Sprintf("%s %s%s", formatCode(i.PackageName, lintCtx.Cfg), msgSuffix, userSuppliedMsgSuffix),
|
||
|
FromLinter: linterName,
|
||
|
}, pass))
|
||
|
}
|
||
|
mu.Lock()
|
||
|
resIssues = append(resIssues, res...)
|
||
|
mu.Unlock()
|
||
|
|
||
|
return nil, nil
|
||
|
}
|
||
|
}).WithIssuesReporter(func(*linter.Context) []goanalysis.Issue {
|
||
|
return resIssues
|
||
|
}).WithLoadMode(goanalysis.LoadModeTypesInfo)
|
||
|
}
|