From b760cb024fe966a725249b55ed9202e5d4850b5b Mon Sep 17 00:00:00 2001 From: Chmouel Boudjnah Date: Wed, 27 Mar 2024 12:33:35 +0100 Subject: [PATCH] Add support for GitOps comments on Push commits Add supports for GitOps comments on Push commits when using the Git Provider. Signed-off-by: Chmouel Boudjnah --- docs/content/docs/guide/authoringprs.md | 4 +- docs/content/docs/guide/running.md | 4 +- pkg/pipelineascode/match.go | 1 - pkg/provider/github/detect.go | 2 +- pkg/provider/github/detect_test.go | 4 +- pkg/provider/github/parse_payload.go | 4 +- test/github_push_retest_test.go | 63 ++++++++++++++++++++++++- test/pkg/github/pr.go | 61 ++++++++++++++++-------- test/pkg/github/setup.go | 1 + 9 files changed, 114 insertions(+), 30 deletions(-) diff --git a/docs/content/docs/guide/authoringprs.md b/docs/content/docs/guide/authoringprs.md index 194d6cf62..e9f0198cb 100644 --- a/docs/content/docs/guide/authoringprs.md +++ b/docs/content/docs/guide/authoringprs.md @@ -173,9 +173,9 @@ You can find more information about the CEL language spec here : ### Matching a PipelineRun on a regexp in a comment -{{< tech_preview "Matching PipelineRun on regexp in comments" >}} +{{< tech_preview "Matching PipelineRun with regexp on comments" >}} -You can match a PipelineRun on a comment on a Pull Request with the annotation +You can match a PipelineRun on a comment on a Pull Request or a [Pushed Commit]({{< relref "/docs/guide/running/#gitops-commands-on-pushed-commits" >}}) with the annotation `pipelinesascode.tekton.dev/on-comment`. The comment is a regexp and if a newly created comment has this regexp it will diff --git a/docs/content/docs/guide/running.md b/docs/content/docs/guide/running.md index 283cc7917..3e26074a9 100644 --- a/docs/content/docs/guide/running.md +++ b/docs/content/docs/guide/running.md @@ -193,7 +193,9 @@ To issue a `GitOps` comment on a pushed commit you can follow these steps: ![GitOps Commits For Comments](/images/gitops-comments-on-commit.png) -Please note that this feature is supported for the GitHub provider only. +{{< hint info >}} +Please note that this feature is only supported when using the GitHub provider. +{{< /hint >}} ### GitOps commands on non-matching PipelineRun diff --git a/pkg/pipelineascode/match.go b/pkg/pipelineascode/match.go index d525da5d2..69f9485b9 100644 --- a/pkg/pipelineascode/match.go +++ b/pkg/pipelineascode/match.go @@ -181,7 +181,6 @@ func (p *PacRun) getPipelineRunsFromRepo(ctx context.Context, repo *v1alpha1.Rep p.eventEmitter.EmitMessage(nil, zap.InfoLevel, "RepositoryCannotLocatePipelineRun", msg) return nil, nil } - pipelineRuns, err = resolve.MetadataResolve(pipelineRuns) if err != nil && len(pipelineRuns) == 0 { p.eventEmitter.EmitMessage(repo, zap.ErrorLevel, "FailedToResolvePipelineRunMetadata", err.Error()) diff --git a/pkg/provider/github/detect.go b/pkg/provider/github/detect.go index e10a106e8..dc882cb01 100644 --- a/pkg/provider/github/detect.go +++ b/pkg/provider/github/detect.go @@ -100,7 +100,7 @@ func detectTriggerTypeFromPayload(ghEventType string, eventInt any) (triggertype return triggertype.Cancel, "" } } - return "", fmt.Sprintf("commit_comment: unsupported action \"%s\"", event.GetAction()) + return triggertype.Comment, "" } return "", fmt.Sprintf("github: event \"%v\" is not supported", ghEventType) } diff --git a/pkg/provider/github/detect_test.go b/pkg/provider/github/detect_test.go index 19947643e..a85524d63 100644 --- a/pkg/provider/github/detect_test.go +++ b/pkg/provider/github/detect_test.go @@ -69,13 +69,13 @@ func TestProvider_Detect(t *testing.T) { processReq: false, }, { - name: "invalid commit_comment Event", + name: "non standard commit_comment Event", event: github.CommitCommentEvent{ Action: github.String("something"), }, eventType: "commit_comment", isGH: true, - processReq: false, + processReq: true, }, { name: "invalid check run Event", diff --git a/pkg/provider/github/parse_payload.go b/pkg/provider/github/parse_payload.go index 030419a57..49256f4a2 100644 --- a/pkg/provider/github/parse_payload.go +++ b/pkg/provider/github/parse_payload.go @@ -395,8 +395,8 @@ func (v *Provider) handleCommitCommentEvent(ctx context.Context, event *github.C runevent.SHA = event.GetComment().GetCommitID() runevent.HeadURL = runevent.URL runevent.BaseURL = runevent.HeadURL - runevent.EventType = "push" - runevent.TriggerTarget = "push" + runevent.TriggerTarget = triggertype.Push + opscomments.SetEventTypeAndTargetPR(runevent, event.GetComment().GetBody()) // Set main as default branch to runevent.HeadBranch, runevent.BaseBranch runevent.HeadBranch, runevent.BaseBranch = "main", "main" diff --git a/test/github_push_retest_test.go b/test/github_push_retest_test.go index 5abd7f618..5a8e0cd7a 100644 --- a/test/github_push_retest_test.go +++ b/test/github_push_retest_test.go @@ -5,12 +5,14 @@ package test import ( "context" + "fmt" "regexp" "testing" "github.com/google/go-github/v59/github" "github.com/openshift-pipelines/pipelines-as-code/test/pkg/cctx" tgithub "github.com/openshift-pipelines/pipelines-as-code/test/pkg/github" + "github.com/openshift-pipelines/pipelines-as-code/test/pkg/options" twait "github.com/openshift-pipelines/pipelines-as-code/test/pkg/wait" tektonv1 "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1" "gotest.tools/v3/assert" @@ -18,11 +20,68 @@ import ( metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) +func TestGithubPushRequestGitOpsCommentOnComment(t *testing.T) { + opsComment := "/hello-world" + ctx := context.Background() + g := &tgithub.PRTest{ + Label: "Github GitOps push/retest request", + YamlFiles: []string{"testdata/pipelinerun-on-comment-annotation.yaml"}, + NoStatusCheck: true, + TargetRefName: options.MainBranch, + } + g.RunPushRequest(ctx, t) + defer g.TearDown(ctx, t) + + // let's make sure we didn't create any PipelineRuns since we only match on-comment here + pruns, err := g.Cnx.Clients.Tekton.TektonV1().PipelineRuns(g.TargetNamespace).List(ctx, metav1.ListOptions{}) + assert.NilError(t, err) + assert.Equal(t, len(pruns.Items), 0) + + g.Cnx.Clients.Log.Infof("Running ops comment %s as Push comment", opsComment) + _, _, err = g.Provider.Client.Repositories.CreateComment(ctx, + g.Options.Organization, + g.Options.Repo, g.SHA, + &github.RepositoryComment{Body: github.String(opsComment)}) + assert.NilError(t, err) + + waitOpts := twait.Opts{ + RepoName: g.TargetNamespace, + Namespace: g.TargetNamespace, + MinNumberStatus: len(g.YamlFiles), + PollTimeout: twait.DefaultTimeout, + TargetSHA: g.SHA, + } + g.Cnx.Clients.Log.Info("Waiting for Repository to be updated") + _, err = twait.UntilRepositoryUpdated(ctx, g.Cnx.Clients, waitOpts) + assert.NilError(t, err) + + g.Cnx.Clients.Log.Infof("Check if we have the repository set as succeeded") + repo, err := g.Cnx.Clients.PipelineAsCode.PipelinesascodeV1alpha1().Repositories(g.TargetNamespace).Get(ctx, g.TargetNamespace, metav1.GetOptions{}) + assert.NilError(t, err) + assert.Equal(t, repo.Status[len(repo.Status)-1].Conditions[0].Status, corev1.ConditionTrue) + + pruns, err = g.Cnx.Clients.Tekton.TektonV1().PipelineRuns(g.TargetNamespace).List(ctx, metav1.ListOptions{}) + assert.NilError(t, err) + assert.Equal(t, len(pruns.Items), len(g.YamlFiles)) + lastPrName := pruns.Items[0].GetName() + err = twait.RegexpMatchingInPodLog( + context.Background(), + g.Cnx, + g.TargetNamespace, + fmt.Sprintf("tekton.dev/pipelineRun=%s", lastPrName), + "step-task", + *regexp.MustCompile(opsComment), + 2) + assert.NilError(t, err) +} + func TestGithubPushRequestGitOpsCommentRetest(t *testing.T) { ctx := context.Background() g := &tgithub.PRTest{ - Label: "Github GitOps push/retest request", - YamlFiles: []string{"testdata/pipelinerun-on-push.yaml", "testdata/pipelinerun.yaml"}, + Label: "Github GitOps push/retest request", + YamlFiles: []string{ + "testdata/pipelinerun-on-push.yaml", "testdata/pipelinerun.yaml", + }, } g.RunPushRequest(ctx, t) defer g.TearDown(ctx, t) diff --git a/test/pkg/github/pr.go b/test/pkg/github/pr.go index 46129f45b..a09e0affd 100644 --- a/test/pkg/github/pr.go +++ b/test/pkg/github/pr.go @@ -16,6 +16,7 @@ import ( "github.com/openshift-pipelines/pipelines-as-code/test/pkg/options" "github.com/openshift-pipelines/pipelines-as-code/test/pkg/payload" "github.com/openshift-pipelines/pipelines-as-code/test/pkg/repository" + "github.com/openshift-pipelines/pipelines-as-code/test/pkg/scm" "github.com/openshift-pipelines/pipelines-as-code/test/pkg/wait" "github.com/tektoncd/pipeline/pkg/names" "go.uber.org/zap" @@ -72,7 +73,7 @@ func PushFilesToRef(ctx context.Context, client *ghlib.Client, commitMessage, ba }, }, &ghlib.CreateCommitOptions{}) if err != nil { - return "", nil, err + return "", nil, fmt.Errorf("error creating commit: %w", err) } ref := &ghlib.Reference{ @@ -83,9 +84,8 @@ func PushFilesToRef(ctx context.Context, client *ghlib.Client, commitMessage, ba } vref, _, err := client.Git.CreateRef(ctx, owner, repo, ref) if err != nil { - return "", nil, err + return "", nil, fmt.Errorf("error creating ref: %w", err) } - return commit.GetSHA(), vref, nil } @@ -151,9 +151,10 @@ func (g *PRTest) RunPullRequest(ctx context.Context, t *testing.T) { targetRefName := fmt.Sprintf("refs/heads/%s", names.SimpleNameGenerator.RestrictLengthWithRandomSuffix("pac-e2e-test")) - sha, vref, err := PushFilesToRef(ctx, ghcnx.Client, logmsg, repoinfo.GetDefaultBranch(), targetRefName, - opts.Organization, opts.Repo, entries) + // TODO(chmouel): Use scm.PushFilesToRefGit instead of API like push + sha, vref, err := PushFilesToRef(ctx, ghcnx.Client, logmsg, repoinfo.GetDefaultBranch(), targetRefName, opts.Organization, opts.Repo, entries) assert.NilError(t, err) + g.Logger.Infof("Commit %s has been created and pushed to %s", sha, vref.GetURL()) number, err := PRCreate(ctx, runcnx, ghcnx, opts.Organization, opts.Repo, targetRefName, repoinfo.GetDefaultBranch(), logmsg) @@ -196,14 +197,21 @@ func (g *PRTest) TearDown(ctx context.Context, t *testing.T) { if g.TargetNamespace != "" { repository.NSTearDown(ctx, t, g.Cnx, g.TargetNamespace) } - g.Logger.Infof("Deleting Ref %s", g.TargetRefName) - _, err := g.Provider.Client.Git.DeleteRef(ctx, g.Options.Organization, g.Options.Repo, g.TargetRefName) - assert.NilError(t, err) + if g.TargetRefName != options.MainBranch { + branch := fmt.Sprintf("heads/%s", filepath.Base(g.TargetRefName)) + g.Logger.Infof("Deleting Ref %s", branch) + _, err := g.Provider.Client.Git.DeleteRef(ctx, g.Options.Organization, g.Options.Repo, branch) + assert.NilError(t, err) + } } func (g *PRTest) RunPushRequest(ctx context.Context, t *testing.T) { targetNS := names.SimpleNameGenerator.RestrictLengthWithRandomSuffix("pac-e2e-push") - targetBranch := targetNS + + targetBranch := g.TargetRefName + if targetBranch == "" { + targetBranch = targetNS + } targetEvent := "push" ctx, runcnx, opts, ghcnx, err := Setup(ctx, g.SecondController, g.Webhook) assert.NilError(t, err) @@ -234,19 +242,34 @@ func (g *PRTest) RunPushRequest(ctx context.Context, t *testing.T) { targetNS, targetBranch, targetEvent, map[string]string{}) assert.NilError(t, err) - targetRefName := fmt.Sprintf("refs/heads/%s", targetBranch) - sha, vref, err := PushFilesToRef(ctx, ghcnx.Client, logmsg, repoinfo.GetDefaultBranch(), targetRefName, opts.Organization, opts.Repo, entries) - g.Logger.Infof("Commit %s has been created and pushed to %s", sha, vref.GetURL()) + targetRefName := targetBranch + cloneURL, err := scm.MakeGitCloneURL(repoinfo.GetCloneURL(), "git", *ghcnx.Token) + assert.NilError(t, err) + scmOpts := scm.Opts{ + GitURL: cloneURL, + TargetRefName: targetRefName, + BaseRefName: repoinfo.GetDefaultBranch(), + WebURL: repoinfo.GetHTMLURL(), + Log: runcnx.Clients.Log, + CommitTitle: logmsg, + } + scm.PushFilesToRefGit(t, &scmOpts, entries) + branch, _, err := ghcnx.Client.Repositories.GetBranch(ctx, opts.Organization, opts.Repo, targetBranch, 1) + assert.NilError(t, err) + sha := branch.GetCommit().GetSHA() + g.Logger.Infof("Commit %s has been created and pushed to %s in branch %s", sha, branch.GetCommit().GetHTMLURL(), branch.GetName()) assert.NilError(t, err) - sopt := wait.SuccessOpt{ - Title: logmsg, - OnEvent: triggertype.Push.String(), - TargetNS: targetNS, - NumberofPRMatch: len(g.YamlFiles), - SHA: sha, + if !g.NoStatusCheck { + sopt := wait.SuccessOpt{ + Title: logmsg, + OnEvent: triggertype.Push.String(), + TargetNS: targetNS, + NumberofPRMatch: len(g.YamlFiles), + SHA: sha, + } + wait.Succeeded(ctx, t, runcnx, opts, sopt) } - wait.Succeeded(ctx, t, runcnx, opts, sopt) g.Cnx = runcnx g.Options = opts diff --git a/test/pkg/github/setup.go b/test/pkg/github/setup.go index c418b4faf..3b60450c4 100644 --- a/test/pkg/github/setup.go +++ b/test/pkg/github/setup.go @@ -109,6 +109,7 @@ func Setup(ctx context.Context, onSecondController, viaDirectWebhook bool) (cont Token: githubToken, URL: githubURL, } + gprovider.Token = &githubToken // TODO: before PR if err := gprovider.SetClient(ctx, nil, event, nil, nil); err != nil { return ctx, nil, options.E2E{}, github.New(), err