mirror of https://github.com/grafana/grafana.git
[Prototyping] Lint dashboard in pull requests (#97199)
* Make lint issues comment * Make it work * Delete previous comments * Move linting to separate package * Fix issue with refactoring * Update test fixture
This commit is contained in:
parent
a1eb1863e5
commit
48feba66e6
3
go.mod
3
go.mod
|
|
@ -480,6 +480,8 @@ require github.com/openzipkin/zipkin-go v0.4.3 // @grafana/oss-big-tent
|
|||
|
||||
require github.com/google/go-github/v66 v66.0.0 // @grafana/grafana-app-platform-squad
|
||||
|
||||
require github.com/grafana/dashboard-linter v0.0.0-20241106223805-1e7999311752 // @grafana/grafana-app-platform-squad
|
||||
|
||||
require (
|
||||
cloud.google.com/go/longrunning v0.6.0 // indirect
|
||||
github.com/at-wat/mqtt-go v0.19.4 // indirect
|
||||
|
|
@ -501,6 +503,7 @@ require (
|
|||
github.com/emirpasic/gods v1.18.1 // indirect
|
||||
github.com/gammazero/deque v0.2.1 // indirect
|
||||
github.com/golang/geo v0.0.0-20210211234256-740aa86cb551 // indirect
|
||||
github.com/grafana/grafana-foundation-sdk/go v0.0.0-20241101005901-83e3491f2a70 // indirect
|
||||
github.com/grafana/jsonparser v0.0.0-20240425183733-ea80629e1a32 // indirect
|
||||
github.com/grafana/loki/pkg/push v0.0.0-20231124142027-e52380921608 // indirect
|
||||
github.com/grafana/sqlds/v4 v4.1.0 // indirect
|
||||
|
|
|
|||
4
go.sum
4
go.sum
|
|
@ -2302,6 +2302,8 @@ github.com/grafana/cue v0.0.0-20230926092038-971951014e3f h1:TmYAMnqg3d5KYEAaT6P
|
|||
github.com/grafana/cue v0.0.0-20230926092038-971951014e3f/go.mod h1:okjJBHFQFer+a41sAe2SaGm1glWS8oEb6CmJvn5Zdws=
|
||||
github.com/grafana/cuetsy v0.1.11 h1:I3IwBhF+UaQxRM79HnImtrAn8REGdb5M3+C4QrYHoWk=
|
||||
github.com/grafana/cuetsy v0.1.11/go.mod h1:Ix97+CPD8ws9oSSxR3/Lf4ahU1I4Np83kjJmDVnLZvc=
|
||||
github.com/grafana/dashboard-linter v0.0.0-20241106223805-1e7999311752 h1:pIEPQPua6kCkMib/nOn//WLgtiA4urf2RPBViSF6CcE=
|
||||
github.com/grafana/dashboard-linter v0.0.0-20241106223805-1e7999311752/go.mod h1:QRxC8NNAbpGx2vz9/XckJcWXtHU3My57byeudPrf0WU=
|
||||
github.com/grafana/dataplane/examples v0.0.1 h1:K9M5glueWyLoL4//H+EtTQq16lXuHLmOhb6DjSCahzA=
|
||||
github.com/grafana/dataplane/examples v0.0.1/go.mod h1:h5YwY8s407/17XF5/dS8XrUtsTVV2RnuW8+m1Mp46mg=
|
||||
github.com/grafana/dataplane/sdata v0.0.9 h1:AGL1LZnCUG4MnQtnWpBPbQ8ZpptaZs14w6kE/MWfg7s=
|
||||
|
|
@ -2322,6 +2324,8 @@ github.com/grafana/grafana-azure-sdk-go/v2 v2.1.2 h1:fV6IgVtViXcYZ4VqTAMuVBTLuGA
|
|||
github.com/grafana/grafana-azure-sdk-go/v2 v2.1.2/go.mod h1:Cbh94bfL5o6mUSaHFiOkx4r4CRKlo/DJLx4dPL8XrE0=
|
||||
github.com/grafana/grafana-cloud-migration-snapshot v1.3.0 h1:F0O9eTy4jHjEd1Z3/qIza2GdY7PYpTddUeaq9p3NKGU=
|
||||
github.com/grafana/grafana-cloud-migration-snapshot v1.3.0/go.mod h1:bd6Cm06EK0MzRO5ahUpbDz1SxNOKu+fzladbaRPHZPY=
|
||||
github.com/grafana/grafana-foundation-sdk/go v0.0.0-20241101005901-83e3491f2a70 h1:69GI3KsF851YnwYp6zHdsskcGp3ZnGsWc+ve8vMp1mc=
|
||||
github.com/grafana/grafana-foundation-sdk/go v0.0.0-20241101005901-83e3491f2a70/go.mod h1:WtWosval1KCZP9BGa42b8aVoJmVXSg0EvQXi9LDSVZQ=
|
||||
github.com/grafana/grafana-google-sdk-go v0.1.0 h1:LKGY8z2DSxKjYfr2flZsWgTRTZ6HGQbTqewE3JvRaNA=
|
||||
github.com/grafana/grafana-google-sdk-go v0.1.0/go.mod h1:Vo2TKWfDVmNTELBUM+3lkrZvFtBws0qSZdXhQxRdJrE=
|
||||
github.com/grafana/grafana-openapi-client-go v0.0.0-20231213163343-bd475d63fb79 h1:r+mU5bGMzcXCRVAuOrTn54S80qbfVkvTdUJZfSfTNbs=
|
||||
|
|
|
|||
|
|
@ -990,8 +990,6 @@ github.com/grafana/alerting v0.0.0-20240917171353-6c25eb6eff10/go.mod h1:GMLi6d0
|
|||
github.com/grafana/cloudflare-go v0.0.0-20230110200409-c627cf6792f2 h1:qhugDMdQ4Vp68H0tp/0iN17DM2ehRo1rLEdOFe/gB8I=
|
||||
github.com/grafana/cloudflare-go v0.0.0-20230110200409-c627cf6792f2/go.mod h1:w/aiO1POVIeXUQyl0VQSZjl5OAGDTL5aX+4v0RA1tcw=
|
||||
github.com/grafana/cog v0.0.4/go.mod h1:lzetOuhGUl/JaSACiJoHvBokf9/fS6PEFaWZvnQu2vs=
|
||||
github.com/grafana/cog v0.0.5 h1:BCa+10i3KvV+KMSQuxlN1DS9cZEwN+EAFc7ZmXqHxQE=
|
||||
github.com/grafana/cog v0.0.5/go.mod h1:lzetOuhGUl/JaSACiJoHvBokf9/fS6PEFaWZvnQu2vs=
|
||||
github.com/grafana/cuetsy v0.1.10/go.mod h1:Ix97+CPD8ws9oSSxR3/Lf4ahU1I4Np83kjJmDVnLZvc=
|
||||
github.com/grafana/go-gelf/v2 v2.0.1 h1:BOChP0h/jLeD+7F9mL7tq10xVkDG15he3T1zHuQaWak=
|
||||
github.com/grafana/go-gelf/v2 v2.0.1/go.mod h1:lexHie0xzYGwCgiRGcvZ723bSNyNI8ZRD4s0CLobh90=
|
||||
|
|
@ -1519,6 +1517,8 @@ github.com/xdg/scram v0.0.0-20180814205039-7eeb5667e42c h1:u40Z8hqBAAQyv+vATcGgV
|
|||
github.com/xdg/stringprep v1.0.0 h1:d9X0esnoa3dFsV0FG35rAT0RIhYFlPq7MiP+DW89La0=
|
||||
github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f h1:J9EGpcZtP0E/raorCMxlFGSTBrsSlaDGf3jU/qvAE2c=
|
||||
github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
|
||||
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo=
|
||||
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
|
||||
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0=
|
||||
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
|
||||
github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74=
|
||||
|
|
@ -1545,6 +1545,8 @@ github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo
|
|||
github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0=
|
||||
github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b h1:FosyBZYxY34Wul7O/MSKey3txpPYyCqVO5ZyceuQJEI=
|
||||
github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b/go.mod h1:ZRKQfBXbGkpdV6QMzT3rU1kSTAnfu1dO8dPKjYprgj8=
|
||||
github.com/zeitlinger/conflate v0.0.0-20230622100834-279724abda8c h1:PtECnCzGLw8MuQ0tmPRaN5c95ZfNTFZOobvgC6A83zk=
|
||||
github.com/zeitlinger/conflate v0.0.0-20230622100834-279724abda8c/go.mod h1:KsJBt1tGR0Q7u+3T7CLN+zITAI06GiXVi/cgP9Xrpb8=
|
||||
github.com/zenazn/goji v1.0.1 h1:4lbD8Mx2h7IvloP7r2C0D6ltZP6Ufip8Hn0wmSK5LR8=
|
||||
github.com/zenazn/goji v1.0.1/go.mod h1:7S9M489iMyHBNxwZnk9/EHS098H4/F6TATF2mIxtB1Q=
|
||||
github.com/ziutek/mymysql v1.5.4 h1:GB0qdRGsTwQSBVYuVShFBKaXSnSnYYC2d9knnE1LHFs=
|
||||
|
|
|
|||
|
|
@ -63,6 +63,9 @@ type GitHubRepositoryConfig struct {
|
|||
// By default, this is false (i.e. we will not create previews).
|
||||
// This option is a no-op if BranchWorkflow is `false` or default.
|
||||
GenerateDashboardPreviews bool `json:"generateDashboardPreviews,omitempty"`
|
||||
|
||||
// PullRequestLinter enables the dashboard linter for this repository in Pull Requests
|
||||
PullRequestLinter bool `json:"pullRequestLinter,omitempty"`
|
||||
}
|
||||
|
||||
// RepositoryType defines the types of Repository
|
||||
|
|
|
|||
|
|
@ -223,6 +223,13 @@ func schema_pkg_apis_provisioning_v0alpha1_GitHubRepositoryConfig(ref common.Ref
|
|||
Format: "",
|
||||
},
|
||||
},
|
||||
"pullRequestLinter": {
|
||||
SchemaProps: spec.SchemaProps{
|
||||
Description: "PullRequestLinter enables the dashboard linter for this repository in Pull Requests",
|
||||
Type: []string{"boolean"},
|
||||
Format: "",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
|
|
|
|||
|
|
@ -0,0 +1,67 @@
|
|||
package lint
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"sort"
|
||||
|
||||
"github.com/grafana/dashboard-linter/lint"
|
||||
)
|
||||
|
||||
type DashboardLinter struct {
|
||||
rules lint.RuleSet
|
||||
}
|
||||
|
||||
// FIXME: what would be a good place to put all the schema validation?
|
||||
type specData struct {
|
||||
Spec json.RawMessage `json:"spec"`
|
||||
}
|
||||
|
||||
func NewDashboardLinter() *DashboardLinter {
|
||||
// TODO: read rules from configuration and pass to the lint function
|
||||
return &DashboardLinter{rules: lint.NewRuleSet()}
|
||||
}
|
||||
|
||||
func (l *DashboardLinter) Lint(ctx context.Context, fileData []byte) ([]Issue, error) {
|
||||
var data specData
|
||||
if err := json.Unmarshal(fileData, &data); err != nil {
|
||||
return nil, fmt.Errorf("unmarshal file data into spec: %w", err)
|
||||
}
|
||||
|
||||
dashboard, err := lint.NewDashboard(data.Spec)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to parse dashboard with linter: %v", err)
|
||||
}
|
||||
|
||||
results, err := l.rules.Lint([]lint.Dashboard{dashboard})
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to lint dashboard: %v", err)
|
||||
}
|
||||
|
||||
byRule := results.ByRule()
|
||||
rules := make([]string, 0, len(byRule))
|
||||
for r := range byRule {
|
||||
rules = append(rules, r)
|
||||
}
|
||||
sort.Strings(rules)
|
||||
|
||||
issues := make([]Issue, 0)
|
||||
for _, rule := range rules {
|
||||
for _, rr := range byRule[rule] {
|
||||
for _, r := range rr.Result.Results {
|
||||
if r.Severity != lint.Error && r.Severity != lint.Warning {
|
||||
continue
|
||||
}
|
||||
|
||||
issues = append(issues, Issue{
|
||||
Rule: rule,
|
||||
Severity: r.Severity,
|
||||
Message: r.Message,
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return issues, nil
|
||||
}
|
||||
|
|
@ -0,0 +1,17 @@
|
|||
package lint
|
||||
|
||||
import (
|
||||
"context"
|
||||
|
||||
"github.com/grafana/dashboard-linter/lint"
|
||||
)
|
||||
|
||||
type Issue struct {
|
||||
Severity lint.Severity
|
||||
Rule string
|
||||
Message string
|
||||
}
|
||||
|
||||
type Linter interface {
|
||||
Lint(ctx context.Context, data []byte) ([]Issue, error)
|
||||
}
|
||||
|
|
@ -30,6 +30,7 @@ import (
|
|||
provisioning "github.com/grafana/grafana/pkg/apis/provisioning/v0alpha1"
|
||||
grafanaregistry "github.com/grafana/grafana/pkg/apiserver/registry/generic"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/auth"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/lint"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/repository"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/repository/github"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/resources"
|
||||
|
|
@ -214,7 +215,8 @@ func (b *ProvisioningAPIBuilder) asRepository(ctx context.Context, obj runtime.O
|
|||
return nil, fmt.Errorf("invalid base URL: %w", err)
|
||||
}
|
||||
|
||||
return repository.NewGitHub(ctx, r, b.ghFactory, baseURL), nil
|
||||
linter := lint.NewDashboardLinter()
|
||||
return repository.NewGitHub(ctx, r, b.ghFactory, baseURL, linter), nil
|
||||
case provisioning.S3RepositoryType:
|
||||
return repository.NewS3(r), nil
|
||||
default:
|
||||
|
|
|
|||
|
|
@ -20,6 +20,7 @@ import (
|
|||
"k8s.io/apiserver/pkg/registry/rest"
|
||||
|
||||
provisioning "github.com/grafana/grafana/pkg/apis/provisioning/v0alpha1"
|
||||
"github.com/grafana/grafana/pkg/registry/apis/provisioning/lint"
|
||||
pgh "github.com/grafana/grafana/pkg/registry/apis/provisioning/repository/github"
|
||||
)
|
||||
|
||||
|
|
@ -28,6 +29,7 @@ type githubRepository struct {
|
|||
config *provisioning.Repository
|
||||
gh pgh.Client
|
||||
baseURL *url.URL
|
||||
linter lint.Linter
|
||||
}
|
||||
|
||||
var _ Repository = (*githubRepository)(nil)
|
||||
|
|
@ -37,12 +39,14 @@ func NewGitHub(
|
|||
config *provisioning.Repository,
|
||||
factory pgh.ClientFactory,
|
||||
baseURL *url.URL,
|
||||
linter lint.Linter,
|
||||
) *githubRepository {
|
||||
return &githubRepository{
|
||||
config: config,
|
||||
logger: slog.Default().With("logger", "github-repository"),
|
||||
gh: factory.New(ctx, config.Spec.GitHub.Token),
|
||||
baseURL: baseURL,
|
||||
linter: linter,
|
||||
}
|
||||
}
|
||||
|
||||
|
|
@ -484,18 +488,8 @@ func (r *githubRepository) onPushEvent(ctx context.Context, logger *slog.Logger,
|
|||
return nil
|
||||
}
|
||||
|
||||
const commentTemplate = `Hey there! 🎉
|
||||
Grafana spotted some changes for your resources in this pull request:
|
||||
|
||||
| File Name | Type | Path | Action | Links |
|
||||
|-----------|------|------|--------|-------|
|
||||
{{- range .}}
|
||||
| {{.Filename}} | {{.Type}} | {{.Path}} | {{.Action}} | {{if .Original}}[Original]({{.Original}}){{end}}{{if .Current}}, [Current]({{.Current}}){{end}}{{if .Preview}}, [Preview]({{.Preview}}){{end}}|
|
||||
{{- end}}
|
||||
|
||||
Click the preview links above to view how your changes will look and compare them with the original and current versions.`
|
||||
|
||||
type commentFile struct {
|
||||
// changedResource represents a resource that has changed in a pull request.
|
||||
type changedResource struct {
|
||||
Filename string
|
||||
Path string
|
||||
Action string
|
||||
|
|
@ -503,13 +497,27 @@ type commentFile struct {
|
|||
Original string
|
||||
Current string
|
||||
Preview string
|
||||
Data []byte
|
||||
}
|
||||
|
||||
// onPullRequestEvent is called when a pull request event is received
|
||||
// If the pull request is opened, reponed or synchronize, we read the files changed.
|
||||
func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.Logger, event *github.PullRequestEvent, factory FileReplicatorFactory) error {
|
||||
action := event.GetAction()
|
||||
r.logger.InfoContext(ctx, "processing pull request event", "number", event.GetNumber(), "action", action)
|
||||
logger = logger.With("pull_request", event.GetNumber(), "action", action, "nnumber", event.GetNumber())
|
||||
logger.InfoContext(
|
||||
ctx,
|
||||
"processing pull request event",
|
||||
"linter",
|
||||
r.Config().Spec.GitHub.PullRequestLinter,
|
||||
"previews",
|
||||
r.Config().Spec.GitHub.GenerateDashboardPreviews,
|
||||
)
|
||||
|
||||
if !r.Config().Spec.GitHub.PullRequestLinter && !r.Config().Spec.GitHub.GenerateDashboardPreviews {
|
||||
logger.DebugContext(ctx, "no action required on pull request event")
|
||||
return nil
|
||||
}
|
||||
|
||||
if event.GetRepo() == nil {
|
||||
return fmt.Errorf("missing repository in pull request event")
|
||||
|
|
@ -530,7 +538,7 @@ func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.
|
|||
return fmt.Errorf("list pull request files: %w", err)
|
||||
}
|
||||
|
||||
rows := make([]commentFile, 0)
|
||||
changedResources := make([]changedResource, 0)
|
||||
|
||||
baseBranch := event.GetPullRequest().GetBase().GetRef()
|
||||
mainBranch := r.config.Spec.GitHub.Branch
|
||||
|
|
@ -542,9 +550,8 @@ func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.
|
|||
|
||||
prURL := event.GetPullRequest().GetHTMLURL()
|
||||
|
||||
// TODO: implement the real handling of the files
|
||||
for _, file := range files {
|
||||
row := commentFile{
|
||||
resource := changedResource{
|
||||
Filename: path.Base(file.GetFilename()),
|
||||
Path: file.GetFilename(),
|
||||
Action: file.GetStatus(),
|
||||
|
|
@ -558,23 +565,23 @@ func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.
|
|||
// reference: https://docs.github.com/en/rest/commits/commits?apiVersion=2022-11-28#get-a-commit
|
||||
switch file.GetStatus() {
|
||||
case "added":
|
||||
row.Preview = r.previewURL(ref, path, prURL)
|
||||
resource.Preview = r.previewURL(ref, path, prURL)
|
||||
case "modified":
|
||||
row.Original = r.previewURL(baseBranch, path, prURL)
|
||||
row.Current = r.previewURL(mainBranch, path, prURL)
|
||||
row.Preview = r.previewURL(ref, path, prURL)
|
||||
resource.Original = r.previewURL(baseBranch, path, prURL)
|
||||
resource.Current = r.previewURL(mainBranch, path, prURL)
|
||||
resource.Preview = r.previewURL(ref, path, prURL)
|
||||
case "removed":
|
||||
row.Original = r.previewURL(baseBranch, path, prURL)
|
||||
row.Current = r.previewURL(mainBranch, path, prURL)
|
||||
resource.Original = r.previewURL(baseBranch, path, prURL)
|
||||
resource.Current = r.previewURL(mainBranch, path, prURL)
|
||||
ref = baseBranch
|
||||
case "renamed":
|
||||
row.Original = r.previewURL(baseBranch, file.GetPreviousFilename(), prURL)
|
||||
row.Current = r.previewURL(mainBranch, file.GetPreviousFilename(), prURL)
|
||||
row.Preview = r.previewURL(ref, path, prURL)
|
||||
resource.Original = r.previewURL(baseBranch, file.GetPreviousFilename(), prURL)
|
||||
resource.Current = r.previewURL(mainBranch, file.GetPreviousFilename(), prURL)
|
||||
resource.Preview = r.previewURL(ref, path, prURL)
|
||||
case "changed":
|
||||
row.Original = r.previewURL(baseBranch, path, prURL)
|
||||
row.Current = r.previewURL(mainBranch, path, prURL)
|
||||
row.Preview = r.previewURL(ref, path, prURL)
|
||||
resource.Original = r.previewURL(baseBranch, path, prURL)
|
||||
resource.Current = r.previewURL(mainBranch, path, prURL)
|
||||
resource.Preview = r.previewURL(ref, path, prURL)
|
||||
case "unchanged":
|
||||
logger.InfoContext(ctx, "ignore unchanged file")
|
||||
continue
|
||||
|
|
@ -589,6 +596,7 @@ func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.
|
|||
continue
|
||||
}
|
||||
|
||||
// TODO: how does this validation works vs linting?
|
||||
ok, err := replicator.Validate(ctx, f)
|
||||
if err != nil {
|
||||
logger.ErrorContext(ctx, "failed to validate file", "error", err)
|
||||
|
|
@ -599,34 +607,133 @@ func (r *githubRepository) onPullRequestEvent(ctx context.Context, logger *slog.
|
|||
continue
|
||||
}
|
||||
|
||||
resource.Data = f.Data
|
||||
logger.InfoContext(ctx, "resource changed")
|
||||
rows = append(rows, row)
|
||||
changedResources = append(changedResources, resource)
|
||||
}
|
||||
|
||||
if len(rows) == 0 {
|
||||
if err := r.previewPullRequest(ctx, event.GetNumber(), changedResources); err != nil {
|
||||
logger.ErrorContext(ctx, "failed to comment previews", "error", err)
|
||||
}
|
||||
|
||||
headSha := event.GetPullRequest().GetHead().GetSHA()
|
||||
if err := r.lintPullRequest(ctx, logger, event.GetNumber(), headSha, changedResources); err != nil {
|
||||
logger.ErrorContext(ctx, "failed to lint pull request resources", "error", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
var lintDashboardIssuesTemplate = `Hey there! 👋
|
||||
Grafana found some linting issues in this dashboard you may want to check:
|
||||
{{ range .}}
|
||||
{{ if eq .Severity 4 }}❌{{ else if eq .Severity 3 }}⚠️ {{ end }} [dashboard-linter/{{ .Rule }}](https://github.com/grafana/dashboard-linter/blob/main/docs/rules/{{ .Rule }}.md): {{ .Message }}.
|
||||
{{- end }}
|
||||
`
|
||||
|
||||
// lintPullRequest lints the files changed in the pull request and comments the issues found.
|
||||
// The linter is disabled if the configuration does not have PullRequestLinter enabled.
|
||||
// The only supported type of file to lint is a dashboard.
|
||||
func (r *githubRepository) lintPullRequest(ctx context.Context, logger *slog.Logger, prNumber int, ref string, resources []changedResource) error {
|
||||
if !r.Config().Spec.GitHub.PullRequestLinter {
|
||||
return nil
|
||||
}
|
||||
|
||||
tmpl, err := template.New("comment").Parse(commentTemplate)
|
||||
logger.InfoContext(ctx, "lint pull request")
|
||||
|
||||
// Clear all previous comments because we don't know if the files have changed
|
||||
if err := r.gh.ClearAllPullRequestFileComments(ctx, r.config.Spec.GitHub.Owner, r.config.Spec.GitHub.Repository, prNumber); err != nil {
|
||||
return fmt.Errorf("clear pull request comments: %w", err)
|
||||
}
|
||||
|
||||
for _, resource := range resources {
|
||||
if resource.Action == "removed" || resource.Type != "dashboard" {
|
||||
continue
|
||||
}
|
||||
|
||||
logger := logger.With("file", resource.Path)
|
||||
if err := r.lintPullRequestFile(ctx, logger, prNumber, ref, resource); err != nil {
|
||||
logger.ErrorContext(ctx, "failed to lint file", "error", err)
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// lintPullRequestFile lints a file and comments the issues found.
|
||||
func (r *githubRepository) lintPullRequestFile(ctx context.Context, logger *slog.Logger, prNumber int, ref string, resource changedResource) error {
|
||||
issues, err := r.linter.Lint(ctx, resource.Data)
|
||||
if err != nil {
|
||||
return fmt.Errorf("lint file: %w", err)
|
||||
}
|
||||
|
||||
if len(issues) == 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
// FIXME: we should not be compiling this all the time
|
||||
tmpl, err := template.New("comment").Parse(lintDashboardIssuesTemplate)
|
||||
if err != nil {
|
||||
return fmt.Errorf("parse lint comment template: %w", err)
|
||||
}
|
||||
|
||||
var buf bytes.Buffer
|
||||
if err := tmpl.Execute(&buf, issues); err != nil {
|
||||
return fmt.Errorf("execute lint comment template: %w", err)
|
||||
}
|
||||
|
||||
comment := pgh.FileComment{
|
||||
Content: buf.String(),
|
||||
Path: resource.Path,
|
||||
Position: 1, // create a top-level comment
|
||||
Ref: ref,
|
||||
}
|
||||
|
||||
// FIXME: comment with Grafana Logo
|
||||
// FIXME: comment author should be written by Grafana and not the user
|
||||
if err := r.gh.CreatePullRequestFileComment(ctx, r.config.Spec.GitHub.Owner, r.config.Spec.GitHub.Repository, prNumber, comment); err != nil {
|
||||
return fmt.Errorf("create pull request comment: %w", err)
|
||||
}
|
||||
logger.InfoContext(ctx, "lint comment created", "issues", len(issues))
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
const previewsCommentTemplate = `Hey there! 🎉
|
||||
Grafana spotted some changes for your resources in this pull request:
|
||||
|
||||
| File Name | Type | Path | Action | Links |
|
||||
|-----------|------|------|--------|-------|
|
||||
{{- range .}}
|
||||
| {{.Filename}} | {{.Type}} | {{.Path}} | {{.Action}} | {{if .Original}}[Original]({{.Original}}){{end}}{{if .Current}}, [Current]({{.Current}}){{end}}{{if .Preview}}, [Preview]({{.Preview}}){{end}}|
|
||||
{{- end}}
|
||||
|
||||
Click the preview links above to view how your changes will look and compare them with the original and current versions.`
|
||||
|
||||
func (r *githubRepository) previewPullRequest(ctx context.Context, prNumber int, resources []changedResource) error {
|
||||
if !r.Config().Spec.GitHub.GenerateDashboardPreviews || len(resources) == 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
// FIXME: we should not be compiling this all the time
|
||||
tmpl, err := template.New("comment").Parse(previewsCommentTemplate)
|
||||
if err != nil {
|
||||
return fmt.Errorf("parse comment template: %w", err)
|
||||
}
|
||||
|
||||
var buf bytes.Buffer
|
||||
if err := tmpl.Execute(&buf, rows); err != nil {
|
||||
if err := tmpl.Execute(&buf, resources); err != nil {
|
||||
return fmt.Errorf("execute comment template: %w", err)
|
||||
}
|
||||
|
||||
comment := buf.String()
|
||||
|
||||
// TODO: comment with Grafana Logo
|
||||
// TODO: comment author should be written by Grafana and not the user
|
||||
if err := r.gh.CreatePullRequestComment(ctx, r.config.Spec.GitHub.Owner, r.config.Spec.GitHub.Repository, event.GetNumber(), comment); err != nil {
|
||||
// FIXME: comment with Grafana Logo
|
||||
// FIXME: comment author should be written by Grafana and not the user
|
||||
if err := r.gh.CreatePullRequestComment(ctx, r.config.Spec.GitHub.Owner, r.config.Spec.GitHub.Repository, prNumber, comment); err != nil {
|
||||
return fmt.Errorf("create pull request comment: %w", err)
|
||||
}
|
||||
|
||||
r.logger.InfoContext(ctx, "comment created", "pull_request", event.GetNumber(), "num_changes", len(rows))
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -78,6 +78,8 @@ type Client interface {
|
|||
|
||||
ListPullRequestFiles(ctx context.Context, owner, repository string, number int) ([]CommitFile, error)
|
||||
CreatePullRequestComment(ctx context.Context, owner, repository string, number int, body string) error
|
||||
CreatePullRequestFileComment(ctx context.Context, owner, repository string, number int, comment FileComment) error
|
||||
ClearAllPullRequestFileComments(ctx context.Context, owner, repository string, number int) error
|
||||
}
|
||||
|
||||
type RepositoryContent interface {
|
||||
|
|
@ -106,6 +108,13 @@ type CommitFile interface {
|
|||
GetStatus() string
|
||||
}
|
||||
|
||||
type FileComment struct {
|
||||
Content string
|
||||
Path string
|
||||
Position int
|
||||
Ref string
|
||||
}
|
||||
|
||||
type CreateFileOptions struct {
|
||||
// The message of the commit. May be empty, in which case a default value is entered.
|
||||
Message string
|
||||
|
|
|
|||
|
|
@ -41,6 +41,24 @@ func (_m *MockClient) BranchExists(ctx context.Context, owner string, repository
|
|||
return r0, r1
|
||||
}
|
||||
|
||||
// ClearAllPullRequestFileComments provides a mock function with given fields: ctx, owner, repository, number
|
||||
func (_m *MockClient) ClearAllPullRequestFileComments(ctx context.Context, owner string, repository string, number int) error {
|
||||
ret := _m.Called(ctx, owner, repository, number)
|
||||
|
||||
if len(ret) == 0 {
|
||||
panic("no return value specified for ClearAllPullRequestFileComments")
|
||||
}
|
||||
|
||||
var r0 error
|
||||
if rf, ok := ret.Get(0).(func(context.Context, string, string, int) error); ok {
|
||||
r0 = rf(ctx, owner, repository, number)
|
||||
} else {
|
||||
r0 = ret.Error(0)
|
||||
}
|
||||
|
||||
return r0
|
||||
}
|
||||
|
||||
// CreateBranch provides a mock function with given fields: ctx, owner, repository, sourceBranch, branchName
|
||||
func (_m *MockClient) CreateBranch(ctx context.Context, owner string, repository string, sourceBranch string, branchName string) error {
|
||||
ret := _m.Called(ctx, owner, repository, sourceBranch, branchName)
|
||||
|
|
@ -95,6 +113,24 @@ func (_m *MockClient) CreatePullRequestComment(ctx context.Context, owner string
|
|||
return r0
|
||||
}
|
||||
|
||||
// CreatePullRequestFileComment provides a mock function with given fields: ctx, owner, repository, number, comment
|
||||
func (_m *MockClient) CreatePullRequestFileComment(ctx context.Context, owner string, repository string, number int, comment FileComment) error {
|
||||
ret := _m.Called(ctx, owner, repository, number, comment)
|
||||
|
||||
if len(ret) == 0 {
|
||||
panic("no return value specified for CreatePullRequestFileComment")
|
||||
}
|
||||
|
||||
var r0 error
|
||||
if rf, ok := ret.Get(0).(func(context.Context, string, string, int, FileComment) error); ok {
|
||||
r0 = rf(ctx, owner, repository, number, comment)
|
||||
} else {
|
||||
r0 = ret.Error(0)
|
||||
}
|
||||
|
||||
return r0
|
||||
}
|
||||
|
||||
// CreateWebhook provides a mock function with given fields: ctx, owner, repository, cfg
|
||||
func (_m *MockClient) CreateWebhook(ctx context.Context, owner string, repository string, cfg WebhookConfig) error {
|
||||
ret := _m.Called(ctx, owner, repository, cfg)
|
||||
|
|
|
|||
|
|
@ -344,6 +344,55 @@ func (r *realImpl) CreatePullRequestComment(ctx context.Context, owner, reposito
|
|||
return nil
|
||||
}
|
||||
|
||||
func (r *realImpl) CreatePullRequestFileComment(ctx context.Context, owner, repository string, number int, comment FileComment) error {
|
||||
commentRequest := &github.PullRequestComment{
|
||||
Body: &comment.Content,
|
||||
CommitID: &comment.Ref,
|
||||
Path: &comment.Path,
|
||||
Position: &comment.Position,
|
||||
}
|
||||
|
||||
if _, _, err := r.gh.PullRequests.CreateComment(ctx, owner, repository, number, commentRequest); err != nil {
|
||||
var ghErr *github.ErrorResponse
|
||||
if errors.As(err, &ghErr) && ghErr.Response.StatusCode == http.StatusServiceUnavailable {
|
||||
return ErrServiceUnavailable
|
||||
}
|
||||
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (r *realImpl) ClearAllPullRequestFileComments(ctx context.Context, owner, repository string, number int) error {
|
||||
comments, _, err := r.gh.PullRequests.ListComments(ctx, owner, repository, number, nil)
|
||||
if err != nil {
|
||||
var ghErr *github.ErrorResponse
|
||||
if errors.As(err, &ghErr) && ghErr.Response.StatusCode == http.StatusServiceUnavailable {
|
||||
return ErrServiceUnavailable
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
userLogin, _, err := r.gh.Users.Get(ctx, "")
|
||||
if err != nil {
|
||||
return fmt.Errorf("get user: %w", err)
|
||||
}
|
||||
|
||||
for _, c := range comments {
|
||||
// skip if comments were not created by us
|
||||
if c.User.GetLogin() != userLogin.GetLogin() {
|
||||
continue
|
||||
}
|
||||
|
||||
if _, err := r.gh.PullRequests.DeleteComment(ctx, owner, repository, c.GetID()); err != nil {
|
||||
return fmt.Errorf("delete comment: %w", err)
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
type realRepositoryContent struct {
|
||||
real *github.RepositoryContent
|
||||
}
|
||||
|
|
|
|||
|
|
@ -205,6 +205,7 @@ func TestIntegrationProvisioning(t *testing.T) {
|
|||
"branchWorkflow": true,
|
||||
"generateDashboardPreviews": true,
|
||||
"owner": "grafana",
|
||||
"pullRequestLinter": true,
|
||||
"repository": "git-ui-sync-demo",
|
||||
"token": "github_pat_dummy",
|
||||
"webhookSecret": "dummyWebhookSecret",
|
||||
|
|
|
|||
|
|
@ -17,5 +17,6 @@ spec:
|
|||
branch: dummy-branch
|
||||
branchWorkflow: true
|
||||
generateDashboardPreviews: true
|
||||
pullRequestLinter: true
|
||||
token: "github_pat_dummy"
|
||||
webhookSecret: "dummyWebhookSecret"
|
||||
Loading…
Reference in New Issue