feat(cli): .glint.yml config and inline suppression comments
Adds project-level configuration and per-job suppression directives:
.glint.yml (searched from pipeline dir up to the git root):
- ignore: [GL007, GL032] — suppress rules globally for the project
- severity: {GL004: warning} — override rule severity (error/warning/ignore)
- stages: [quality] — extra stages beyond the pipeline's stages: block
- token: / url: / cache_dir: — defaults for flags; lower priority than
CLI flags and environment variables
Inline suppression (# glint: ignore):
- Place "# glint: ignore GL007" immediately before a job definition to
suppress that rule for the specific job only
- Multiple rules: "# glint: ignore GL007, GL032" (comma or space separated)
- Wildcard: "# glint: ignore all" suppresses every finding for the job
- Suppressions are scoped to the annotated job; pipeline-level findings
are unaffected
- Parsed from yaml.Node head/line comments in the first parse pass;
stored in Pipeline.Suppressions (root file only, not includes)
New packages: internal/config (Load, walk-up search, .git boundary stop)
New files: cmd/glint/filter.go (applyConfig, isSuppressed helpers)
Tests: config_test.go, parser_suppress_test.go, filter_test.go
Validate fixtures: testdata/config_ignored/, config_severity/, config_suppress/
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -3,6 +3,7 @@ package model
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"strings"
|
||||
|
||||
"gopkg.in/yaml.v3"
|
||||
)
|
||||
@@ -60,6 +61,14 @@ func ParseBytes(data []byte) (*Pipeline, error) {
|
||||
continue
|
||||
}
|
||||
|
||||
// Extract any inline suppression directive from the job's head or line comment.
|
||||
if rules := parseSuppressComment(keyNode.HeadComment, keyNode.LineComment); len(rules) > 0 {
|
||||
if p.Suppressions == nil {
|
||||
p.Suppressions = map[string][]string{}
|
||||
}
|
||||
p.Suppressions[key] = rules
|
||||
}
|
||||
|
||||
var rawMap map[string]any
|
||||
if err := valNode.Decode(&rawMap); err != nil {
|
||||
return nil, fmt.Errorf("parsing raw job %q: %w", key, err)
|
||||
@@ -78,6 +87,39 @@ func ParseBytes(data []byte) (*Pipeline, error) {
|
||||
return p, nil
|
||||
}
|
||||
|
||||
// parseSuppressComment scans head/line comments from a YAML key node for a
|
||||
// "# glint: ignore RULE [RULE ...]" directive. Returns the list of rule IDs
|
||||
// to suppress (uppercased), or []string{"*"} for "# glint: ignore all".
|
||||
// Returns nil when no directive is found.
|
||||
func parseSuppressComment(headComment, lineComment string) []string {
|
||||
for _, raw := range []string{headComment, lineComment} {
|
||||
for _, line := range strings.Split(raw, "\n") {
|
||||
line = strings.TrimLeft(line, "# \t")
|
||||
if !strings.HasPrefix(line, "glint:") {
|
||||
continue
|
||||
}
|
||||
rest := strings.TrimSpace(strings.TrimPrefix(line, "glint:"))
|
||||
if !strings.HasPrefix(rest, "ignore") {
|
||||
continue
|
||||
}
|
||||
rest = strings.TrimSpace(strings.TrimPrefix(rest, "ignore"))
|
||||
if rest == "" || strings.EqualFold(rest, "all") {
|
||||
return []string{"*"}
|
||||
}
|
||||
var rules []string
|
||||
for _, part := range strings.FieldsFunc(rest, func(r rune) bool {
|
||||
return r == ',' || r == ' ' || r == '\t'
|
||||
}) {
|
||||
if p := strings.TrimSpace(part); p != "" {
|
||||
rules = append(rules, strings.ToUpper(p))
|
||||
}
|
||||
}
|
||||
return rules
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// sanitizeYAMLEscapes rewrites double-quoted YAML strings, replacing the \/
|
||||
// escape sequence (unrecognised by gopkg.in/yaml.v3) with \\/ so that the
|
||||
// parser produces a literal backslash+slash — preserving regex patterns like
|
||||
|
||||
Reference in New Issue
Block a user