From 5cbceba8b0919a112de1344077868d350523d32c Mon Sep 17 00:00:00 2001 From: Tymofii Polekhin Date: Sat, 18 Mar 2023 23:43:11 +0100 Subject: [PATCH] fix: allow `Require Linear History` when selecting merge method (#3211) * fix: account for Require Linear History when selecting merge method * fix logic and add tests * add nolint hints for w.Write --------- Co-authored-by: nitrocode <7775707+nitrocode@users.noreply.github.com> --- server/events/vcs/github_client.go | 12 +- server/events/vcs/github_client_test.go | 165 +++++++++++++++--- ...nch-protection-require-linear-history.json | 6 + 3 files changed, 156 insertions(+), 27 deletions(-) create mode 100644 server/events/vcs/testdata/github-branch-protection-require-linear-history.json diff --git a/server/events/vcs/github_client.go b/server/events/vcs/github_client.go index 7287913436..10b31add1e 100644 --- a/server/events/vcs/github_client.go +++ b/server/events/vcs/github_client.go @@ -579,13 +579,23 @@ func (g *GithubClient) MergePull(pull models.PullRequest, pullOptions models.Pul if err != nil { return errors.Wrap(err, "fetching repo info") } + protection, _, err := g.client.Repositories.GetBranchProtection(context.Background(), repo.Owner.GetLogin(), *repo.Name, pull.BaseBranch) + if err != nil { + if !errors.Is(err, github.ErrBranchNotProtected) { + return errors.Wrap(err, "getting branch protection rules") + } + } + requireLinearHistory := false + if protection != nil { + requireLinearHistory = protection.GetRequireLinearHistory().Enabled + } const ( defaultMergeMethod = "merge" rebaseMergeMethod = "rebase" squashMergeMethod = "squash" ) method := defaultMergeMethod - if !repo.GetAllowMergeCommit() { + if !repo.GetAllowMergeCommit() || requireLinearHistory { if repo.GetAllowRebaseMerge() { method = rebaseMergeMethod } else if repo.GetAllowSquashMerge() { diff --git a/server/events/vcs/github_client_test.go b/server/events/vcs/github_client_test.go index c1c30f03d4..3da4ad0b7c 100644 --- a/server/events/vcs/github_client_test.go +++ b/server/events/vcs/github_client_test.go @@ -787,6 +787,10 @@ func TestGithubClient_MergePullHandlesError(t *testing.T) { defer r.Body.Close() // nolint: errcheck w.WriteHeader(c.code) w.Write([]byte(resp)) // nolint: errcheck + case "/api/v3/repos/runatlantis/atlantis/branches/master/protection": + w.WriteHeader(404) + w.Write([]byte("{\"message\":\"Branch not protected\"}")) // nolint: errcheck + return default: t.Errorf("got unexpected request at %q", r.RequestURI) http.Error(w, "not found", http.StatusNotFound) @@ -813,7 +817,8 @@ func TestGithubClient_MergePullHandlesError(t *testing.T) { Hostname: "github.com", }, }, - Num: 1, + Num: 1, + BaseBranch: "master", }, models.PullRequestOptions{ DeleteSourceBranchOnMerge: false, }) @@ -831,40 +836,132 @@ func TestGithubClient_MergePullHandlesError(t *testing.T) { // use that method func TestGithubClient_MergePullCorrectMethod(t *testing.T) { cases := map[string]struct { - allowMerge bool - allowRebase bool - allowSquash bool - expMethod string + allowMerge bool + allowRebase bool + allowSquash bool + requireLinearHistory bool + protectedBranch bool + expMethod string }{ "all true": { - allowMerge: true, - allowRebase: true, - allowSquash: true, - expMethod: "merge", + allowMerge: true, + allowRebase: true, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: false, + expMethod: "merge", }, "all false (edge case)": { - allowMerge: false, - allowRebase: false, - allowSquash: false, - expMethod: "merge", + allowMerge: false, + allowRebase: false, + allowSquash: false, + requireLinearHistory: false, + protectedBranch: false, + expMethod: "merge", }, "merge: false rebase: true squash: true": { - allowMerge: false, - allowRebase: true, - allowSquash: true, - expMethod: "rebase", + allowMerge: false, + allowRebase: true, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: false, + expMethod: "rebase", }, "merge: false rebase: false squash: true": { - allowMerge: false, - allowRebase: false, - allowSquash: true, - expMethod: "squash", + allowMerge: false, + allowRebase: false, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: false, + expMethod: "squash", }, "merge: false rebase: true squash: false": { - allowMerge: false, - allowRebase: true, - allowSquash: false, - expMethod: "rebase", + allowMerge: false, + allowRebase: true, + allowSquash: false, + requireLinearHistory: false, + protectedBranch: false, + expMethod: "rebase", + }, + "protected, all true, rlh: false": { + allowMerge: true, + allowRebase: true, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: true, + expMethod: "merge", + }, + "protected, all false (edge case), rlh: false": { + allowMerge: false, + allowRebase: false, + allowSquash: false, + requireLinearHistory: false, + protectedBranch: true, + expMethod: "merge", + }, + "protected, merge: false rebase: true squash: true, rlh: false": { + allowMerge: false, + allowRebase: true, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: true, + expMethod: "rebase", + }, + "protected, merge: false rebase: false squash: true, rlh: false": { + allowMerge: false, + allowRebase: false, + allowSquash: true, + requireLinearHistory: false, + protectedBranch: true, + expMethod: "squash", + }, + "protected, merge: false rebase: true squash: false, rlh: false": { + allowMerge: false, + allowRebase: true, + allowSquash: false, + requireLinearHistory: false, + protectedBranch: true, + expMethod: "rebase", + }, + "protected, all true, rlh: true": { + allowMerge: true, + allowRebase: true, + allowSquash: true, + requireLinearHistory: true, + protectedBranch: true, + expMethod: "rebase", + }, + "protected, all false (edge case), rlh: true": { + allowMerge: false, + allowRebase: false, + allowSquash: false, + requireLinearHistory: true, + protectedBranch: true, + expMethod: "merge", + }, + "protected, merge: false rebase: true squash: true, rlh: true": { + allowMerge: false, + allowRebase: true, + allowSquash: true, + requireLinearHistory: true, + protectedBranch: true, + expMethod: "rebase", + }, + "protected, merge: false rebase: false squash: true, rlh: true": { + allowMerge: false, + allowRebase: false, + allowSquash: true, + requireLinearHistory: true, + protectedBranch: true, + expMethod: "squash", + }, + "protected, merge: false rebase: true squash: false, rlh: true": { + allowMerge: false, + allowRebase: true, + allowSquash: false, + requireLinearHistory: true, + protectedBranch: true, + expMethod: "rebase", }, } @@ -874,7 +971,10 @@ func TestGithubClient_MergePullCorrectMethod(t *testing.T) { // Modify response. jsBytes, err := os.ReadFile("testdata/github-repo.json") Ok(t, err) + rlhBytes, err := os.ReadFile("testdata/github-branch-protection-require-linear-history.json") + Ok(t, err) resp := string(jsBytes) + protected := string(rlhBytes) resp = strings.Replace(resp, `"allow_squash_merge": true`, fmt.Sprintf(`"allow_squash_merge": %t`, c.allowSquash), @@ -887,6 +987,10 @@ func TestGithubClient_MergePullCorrectMethod(t *testing.T) { `"allow_rebase_merge": true`, fmt.Sprintf(`"allow_rebase_merge": %t`, c.allowRebase), -1) + protected = strings.Replace(protected, + `"enabled": true`, + fmt.Sprintf(`"enabled": %t`, c.requireLinearHistory), + -1) testServer := httptest.NewTLSServer( http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { @@ -894,6 +998,14 @@ func TestGithubClient_MergePullCorrectMethod(t *testing.T) { case "/api/v3/repos/runatlantis/atlantis": w.Write([]byte(resp)) // nolint: errcheck return + case "/api/v3/repos/runatlantis/atlantis/branches/master/protection": + if c.protectedBranch { + w.Write([]byte(protected)) // nolint: errcheck + } else { + w.WriteHeader(404) + w.Write([]byte("{\"message\":\"Branch not protected\"}")) // nolint: errcheck + } + return case "/api/v3/repos/runatlantis/atlantis/pulls/1/merge": body, err := io.ReadAll(r.Body) Ok(t, err) @@ -936,7 +1048,8 @@ func TestGithubClient_MergePullCorrectMethod(t *testing.T) { Hostname: "github.com", }, }, - Num: 1, + Num: 1, + BaseBranch: "master", }, models.PullRequestOptions{ DeleteSourceBranchOnMerge: false, }) diff --git a/server/events/vcs/testdata/github-branch-protection-require-linear-history.json b/server/events/vcs/testdata/github-branch-protection-require-linear-history.json new file mode 100644 index 0000000000..c4d4de34e4 --- /dev/null +++ b/server/events/vcs/testdata/github-branch-protection-require-linear-history.json @@ -0,0 +1,6 @@ +{ + "url": "https://api.github.com/repos/octocat/Hello-World/branches/master/protection", + "required_linear_history": { + "enabled": true + } +} \ No newline at end of file