[Enhancement] Allow admin to merge pr with protected file changes (#12078)
* [Enhancement] Allow admin to merge pr with protected file changes As tilte, show protected message in diff page and merge box. Signed-off-by: a1012112796 <1012112796@qq.com> * remove unused ver * Update options/locale/locale_en-US.ini Co-authored-by: Cirno the Strongest <1447794+CirnoT@users.noreply.github.com> * Add TrN * Apply suggestions from code review * fix lint * Update options/locale/locale_en-US.ini Co-authored-by: zeripath <art27@cantab.net> * Apply suggestions from code review * move pr proteced files check to TestPatch * Call TestPatch when protected branches settings changed * Apply review suggestion @CirnoT * move to service @lunny * slightly restructure routers/private/hook.go Adds a lot of comments and simplifies the logic Signed-off-by: Andrew Thornton <art27@cantab.net> * placate lint Signed-off-by: Andrew Thornton <art27@cantab.net> * skip duplicate protected files check * fix check logic * slight refactor of TestPatch Signed-off-by: Andrew Thornton <art27@cantab.net> * When checking for protected files changes in TestPatch use the temporary repository Signed-off-by: Andrew Thornton <art27@cantab.net> * fix introduced issue with hook Signed-off-by: Andrew Thornton <art27@cantab.net> * Remove the check on PR index being greater than 0 as it unnecessary Signed-off-by: Andrew Thornton <art27@cantab.net> Co-authored-by: techknowlogick <matti@mdranta.net> Co-authored-by: Cirno the Strongest <1447794+CirnoT@users.noreply.github.com> Co-authored-by: zeripath <art27@cantab.net> Co-authored-by: techknowlogick <techknowlogick@gitea.io>
This commit is contained in:
		
							parent
							
								
									da32d0e72a
								
							
						
					
					
						commit
						dfa7291f8f
					
				
					 19 changed files with 464 additions and 185 deletions
				
			
		|  | @ -209,6 +209,38 @@ func (protectBranch *ProtectedBranch) GetProtectedFilePatterns() []glob.Glob { | ||||||
| 	return extarr | 	return extarr | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // MergeBlockedByProtectedFiles returns true if merge is blocked by protected files change
 | ||||||
|  | func (protectBranch *ProtectedBranch) MergeBlockedByProtectedFiles(pr *PullRequest) bool { | ||||||
|  | 	glob := protectBranch.GetProtectedFilePatterns() | ||||||
|  | 	if len(glob) == 0 { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return len(pr.ChangedProtectedFiles) > 0 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // IsProtectedFile return if path is protected
 | ||||||
|  | func (protectBranch *ProtectedBranch) IsProtectedFile(patterns []glob.Glob, path string) bool { | ||||||
|  | 	if len(patterns) == 0 { | ||||||
|  | 		patterns = protectBranch.GetProtectedFilePatterns() | ||||||
|  | 		if len(patterns) == 0 { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lpath := strings.ToLower(strings.TrimSpace(path)) | ||||||
|  | 
 | ||||||
|  | 	r := false | ||||||
|  | 	for _, pat := range patterns { | ||||||
|  | 		if pat.Match(lpath) { | ||||||
|  | 			r = true | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return r | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GetProtectedBranchByRepoID getting protected branch by repo ID
 | // GetProtectedBranchByRepoID getting protected branch by repo ID
 | ||||||
| func GetProtectedBranchByRepoID(repoID int64) ([]*ProtectedBranch, error) { | func GetProtectedBranchByRepoID(repoID int64) ([]*ProtectedBranch, error) { | ||||||
| 	protectedBranches := make([]*ProtectedBranch, 0) | 	protectedBranches := make([]*ProtectedBranch, 0) | ||||||
|  |  | ||||||
|  | @ -244,6 +244,8 @@ var migrations = []Migration{ | ||||||
| 	NewMigration("add Team review request support", addTeamReviewRequestSupport), | 	NewMigration("add Team review request support", addTeamReviewRequestSupport), | ||||||
| 	// v154 > v155
 | 	// v154 > v155
 | ||||||
| 	NewMigration("add timestamps to Star, Label, Follow, Watch and Collaboration", addTimeStamps), | 	NewMigration("add timestamps to Star, Label, Follow, Watch and Collaboration", addTimeStamps), | ||||||
|  | 	// v155 -> v156
 | ||||||
|  | 	NewMigration("add changed_protected_files column for pull_request table", addChangedProtectedFilesPullRequestColumn), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetCurrentDBVersion returns the current db version
 | // GetCurrentDBVersion returns the current db version
 | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								models/migrations/v155.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								models/migrations/v155.go
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | // Copyright 2020 The Gitea Authors. All rights reserved.
 | ||||||
|  | // Use of this source code is governed by a MIT-style
 | ||||||
|  | // license that can be found in the LICENSE file.
 | ||||||
|  | 
 | ||||||
|  | package migrations | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 
 | ||||||
|  | 	"xorm.io/xorm" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func addChangedProtectedFilesPullRequestColumn(x *xorm.Engine) error { | ||||||
|  | 	type PullRequest struct { | ||||||
|  | 		ChangedProtectedFiles []string `xorm:"TEXT JSON"` | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if err := x.Sync2(new(PullRequest)); err != nil { | ||||||
|  | 		return fmt.Errorf("Sync2: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | @ -45,6 +45,8 @@ type PullRequest struct { | ||||||
| 	CommitsAhead    int | 	CommitsAhead    int | ||||||
| 	CommitsBehind   int | 	CommitsBehind   int | ||||||
| 
 | 
 | ||||||
|  | 	ChangedProtectedFiles []string `xorm:"TEXT JSON"` | ||||||
|  | 
 | ||||||
| 	IssueID int64  `xorm:"INDEX"` | 	IssueID int64  `xorm:"INDEX"` | ||||||
| 	Issue   *Issue `xorm:"-"` | 	Issue   *Issue `xorm:"-"` | ||||||
| 	Index   int64 | 	Index   int64 | ||||||
|  |  | ||||||
|  | @ -123,7 +123,7 @@ func detectEncodingAndBOM(entry *git.TreeEntry, repo *models.Repository) (string | ||||||
| 
 | 
 | ||||||
| // CreateOrUpdateRepoFile adds or updates a file in the given repository
 | // CreateOrUpdateRepoFile adds or updates a file in the given repository
 | ||||||
| func CreateOrUpdateRepoFile(repo *models.Repository, doer *models.User, opts *UpdateRepoFileOptions) (*structs.FileResponse, error) { | func CreateOrUpdateRepoFile(repo *models.Repository, doer *models.User, opts *UpdateRepoFileOptions) (*structs.FileResponse, error) { | ||||||
| 	// If no branch name is set, assume master
 | 	// If no branch name is set, assume default branch
 | ||||||
| 	if opts.OldBranch == "" { | 	if opts.OldBranch == "" { | ||||||
| 		opts.OldBranch = repo.DefaultBranch | 		opts.OldBranch = repo.DefaultBranch | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -1232,6 +1232,8 @@ pulls.required_status_check_administrator = As an administrator, you may still m | ||||||
| pulls.blocked_by_approvals = "This Pull Request doesn't have enough approvals yet. %d of %d approvals granted." | pulls.blocked_by_approvals = "This Pull Request doesn't have enough approvals yet. %d of %d approvals granted." | ||||||
| pulls.blocked_by_rejection = "This Pull Request has changes requested by an official reviewer." | pulls.blocked_by_rejection = "This Pull Request has changes requested by an official reviewer." | ||||||
| pulls.blocked_by_outdated_branch = "This Pull Request is blocked because it's outdated." | pulls.blocked_by_outdated_branch = "This Pull Request is blocked because it's outdated." | ||||||
|  | pulls.blocked_by_changed_protected_files_1= "This Pull Request is blocked because it changes a protected file:" | ||||||
|  | pulls.blocked_by_changed_protected_files_n= "This Pull Request is blocked because it changes protected files:" | ||||||
| pulls.can_auto_merge_desc = This pull request can be merged automatically. | pulls.can_auto_merge_desc = This pull request can be merged automatically. | ||||||
| pulls.cannot_auto_merge_desc = This pull request cannot be merged automatically due to conflicts. | pulls.cannot_auto_merge_desc = This pull request cannot be merged automatically due to conflicts. | ||||||
| pulls.cannot_auto_merge_helper = Merge manually to resolve the conflicts. | pulls.cannot_auto_merge_helper = Merge manually to resolve the conflicts. | ||||||
|  | @ -1779,6 +1781,7 @@ diff.review.comment = Comment | ||||||
| diff.review.approve = Approve | diff.review.approve = Approve | ||||||
| diff.review.reject = Request changes | diff.review.reject = Request changes | ||||||
| diff.committed_by = committed by | diff.committed_by = committed by | ||||||
|  | diff.protected = Protected | ||||||
| 
 | 
 | ||||||
| releases.desc = Track project versions and downloads. | releases.desc = Track project versions and downloads. | ||||||
| release.releases = Releases | release.releases = Releases | ||||||
|  |  | ||||||
|  | @ -16,6 +16,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	repo_module "code.gitea.io/gitea/modules/repository" | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
|  | 	pull_service "code.gitea.io/gitea/services/pull" | ||||||
| 	repo_service "code.gitea.io/gitea/services/repository" | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | @ -545,6 +546,11 @@ func CreateBranchProtection(ctx *context.APIContext, form api.CreateBranchProtec | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err = pull_service.CheckPrsForBaseBranch(ctx.Repo.Repository, protectBranch.BranchName); err != nil { | ||||||
|  | 		ctx.Error(http.StatusInternalServerError, "CheckPrsForBaseBranch", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	// Reload from db to get all whitelists
 | 	// Reload from db to get all whitelists
 | ||||||
| 	bp, err := models.GetProtectedBranchBy(ctx.Repo.Repository.ID, form.BranchName) | 	bp, err := models.GetProtectedBranchBy(ctx.Repo.Repository.ID, form.BranchName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|  | @ -768,6 +774,11 @@ func EditBranchProtection(ctx *context.APIContext, form api.EditBranchProtection | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err = pull_service.CheckPrsForBaseBranch(ctx.Repo.Repository, protectBranch.BranchName); err != nil { | ||||||
|  | 		ctx.Error(http.StatusInternalServerError, "CheckPrsForBaseBranch", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	// Reload from db to ensure get all whitelists
 | 	// Reload from db to ensure get all whitelists
 | ||||||
| 	bp, err := models.GetProtectedBranchBy(repo.ID, bpName) | 	bp, err := models.GetProtectedBranchBy(repo.ID, bpName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|  |  | ||||||
|  | @ -774,7 +774,7 @@ func MergePullRequest(ctx *context.APIContext, form auth.MergePullRequestForm) { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := pull_service.CheckPRReadyToMerge(pr); err != nil { | 	if err := pull_service.CheckPRReadyToMerge(pr, false); err != nil { | ||||||
| 		if !models.IsErrNotAllowedToMerge(err) { | 		if !models.IsErrNotAllowedToMerge(err) { | ||||||
| 			ctx.Error(http.StatusInternalServerError, "CheckPRReadyToMerge", err) | 			ctx.Error(http.StatusInternalServerError, "CheckPRReadyToMerge", err) | ||||||
| 			return | 			return | ||||||
|  |  | ||||||
|  | @ -25,7 +25,6 @@ import ( | ||||||
| 
 | 
 | ||||||
| 	"gitea.com/macaron/macaron" | 	"gitea.com/macaron/macaron" | ||||||
| 	"github.com/go-git/go-git/v5/plumbing" | 	"github.com/go-git/go-git/v5/plumbing" | ||||||
| 	"github.com/gobwas/glob" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func verifyCommits(oldCommitID, newCommitID string, repo *git.Repository, env []string) error { | func verifyCommits(oldCommitID, newCommitID string, repo *git.Repository, env []string) error { | ||||||
|  | @ -59,53 +58,6 @@ func verifyCommits(oldCommitID, newCommitID string, repo *git.Repository, env [] | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func checkFileProtection(oldCommitID, newCommitID string, patterns []glob.Glob, repo *git.Repository, env []string) error { |  | ||||||
| 
 |  | ||||||
| 	stdoutReader, stdoutWriter, err := os.Pipe() |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Error("Unable to create os.Pipe for %s", repo.Path) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer func() { |  | ||||||
| 		_ = stdoutReader.Close() |  | ||||||
| 		_ = stdoutWriter.Close() |  | ||||||
| 	}() |  | ||||||
| 
 |  | ||||||
| 	// This use of ...  is safe as force-pushes have already been ruled out.
 |  | ||||||
| 	err = git.NewCommand("diff", "--name-only", oldCommitID+"..."+newCommitID). |  | ||||||
| 		RunInDirTimeoutEnvFullPipelineFunc(env, -1, repo.Path, |  | ||||||
| 			stdoutWriter, nil, nil, |  | ||||||
| 			func(ctx context.Context, cancel context.CancelFunc) error { |  | ||||||
| 				_ = stdoutWriter.Close() |  | ||||||
| 
 |  | ||||||
| 				scanner := bufio.NewScanner(stdoutReader) |  | ||||||
| 				for scanner.Scan() { |  | ||||||
| 					path := strings.TrimSpace(scanner.Text()) |  | ||||||
| 					if len(path) == 0 { |  | ||||||
| 						continue |  | ||||||
| 					} |  | ||||||
| 					lpath := strings.ToLower(path) |  | ||||||
| 					for _, pat := range patterns { |  | ||||||
| 						if pat.Match(lpath) { |  | ||||||
| 							cancel() |  | ||||||
| 							return models.ErrFilePathProtected{ |  | ||||||
| 								Path: path, |  | ||||||
| 							} |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				if err := scanner.Err(); err != nil { |  | ||||||
| 					return err |  | ||||||
| 				} |  | ||||||
| 				_ = stdoutReader.Close() |  | ||||||
| 				return err |  | ||||||
| 			}) |  | ||||||
| 	if err != nil && !models.IsErrFilePathProtected(err) { |  | ||||||
| 		log.Error("Unable to check file protection for commits from %s to %s in %s: %v", oldCommitID, newCommitID, repo.Path, err) |  | ||||||
| 	} |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func readAndVerifyCommitsFromShaReader(input io.ReadCloser, repo *git.Repository, env []string) error { | func readAndVerifyCommitsFromShaReader(input io.ReadCloser, repo *git.Repository, env []string) error { | ||||||
| 	scanner := bufio.NewScanner(input) | 	scanner := bufio.NewScanner(input) | ||||||
| 	for scanner.Scan() { | 	for scanner.Scan() { | ||||||
|  | @ -202,6 +154,7 @@ func HookPreReceive(ctx *macaron.Context, opts private.HookOptions) { | ||||||
| 			private.GitQuarantinePath+"="+opts.GitQuarantinePath) | 			private.GitQuarantinePath+"="+opts.GitQuarantinePath) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	// Iterate across the provided old commit IDs
 | ||||||
| 	for i := range opts.OldCommitIDs { | 	for i := range opts.OldCommitIDs { | ||||||
| 		oldCommitID := opts.OldCommitIDs[i] | 		oldCommitID := opts.OldCommitIDs[i] | ||||||
| 		newCommitID := opts.NewCommitIDs[i] | 		newCommitID := opts.NewCommitIDs[i] | ||||||
|  | @ -224,146 +177,192 @@ func HookPreReceive(ctx *macaron.Context, opts private.HookOptions) { | ||||||
| 			}) | 			}) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		if protectBranch != nil && protectBranch.IsProtected() { | 
 | ||||||
| 			// detect and prevent deletion
 | 		// Allow pushes to non-protected branches
 | ||||||
| 			if newCommitID == git.EmptySHA { | 		if protectBranch == nil || !protectBranch.IsProtected() { | ||||||
| 				log.Warn("Forbidden: Branch: %s in %-v is protected from deletion", branchName, repo) | 			continue | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// This ref is a protected branch.
 | ||||||
|  | 		//
 | ||||||
|  | 		// First of all we need to enforce absolutely:
 | ||||||
|  | 		//
 | ||||||
|  | 		// 1. Detect and prevent deletion of the branch
 | ||||||
|  | 		if newCommitID == git.EmptySHA { | ||||||
|  | 			log.Warn("Forbidden: Branch: %s in %-v is protected from deletion", branchName, repo) | ||||||
|  | 			ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
|  | 				"err": fmt.Sprintf("branch %s is protected from deletion", branchName), | ||||||
|  | 			}) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// 2. Disallow force pushes to protected branches
 | ||||||
|  | 		if git.EmptySHA != oldCommitID { | ||||||
|  | 			output, err := git.NewCommand("rev-list", "--max-count=1", oldCommitID, "^"+newCommitID).RunInDirWithEnv(repo.RepoPath(), env) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Error("Unable to detect force push between: %s and %s in %-v Error: %v", oldCommitID, newCommitID, repo, err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Fail to detect force push: %v", err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} else if len(output) > 0 { | ||||||
|  | 				log.Warn("Forbidden: Branch: %s in %-v is protected from force push", branchName, repo) | ||||||
| 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
| 					"err": fmt.Sprintf("branch %s is protected from deletion", branchName), | 					"err": fmt.Sprintf("branch %s is protected from force push", branchName), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 
 | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// 3. Enforce require signed commits
 | ||||||
|  | 		if protectBranch.RequireSignedCommits { | ||||||
|  | 			err := verifyCommits(oldCommitID, newCommitID, gitRepo, env) | ||||||
|  | 			if err != nil { | ||||||
|  | 				if !isErrUnverifiedCommit(err) { | ||||||
|  | 					log.Error("Unable to check commits from %s to %s in %-v: %v", oldCommitID, newCommitID, repo, err) | ||||||
|  | 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 						"err": fmt.Sprintf("Unable to check commits from %s to %s: %v", oldCommitID, newCommitID, err), | ||||||
|  | 					}) | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
|  | 				unverifiedCommit := err.(*errUnverifiedCommit).sha | ||||||
|  | 				log.Warn("Forbidden: Branch: %s in %-v is protected from unverified commit %s", branchName, repo, unverifiedCommit) | ||||||
|  | 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("branch %s is protected from unverified commit %s", branchName, unverifiedCommit), | ||||||
| 				}) | 				}) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
|  | 		} | ||||||
| 
 | 
 | ||||||
| 			// detect force push
 | 		// Now there are several tests which can be overridden:
 | ||||||
| 			if git.EmptySHA != oldCommitID { | 		//
 | ||||||
| 				output, err := git.NewCommand("rev-list", "--max-count=1", oldCommitID, "^"+newCommitID).RunInDirWithEnv(repo.RepoPath(), env) | 		// 4. Check protected file patterns - this is overridable from the UI
 | ||||||
| 				if err != nil { | 		changedProtectedfiles := false | ||||||
| 					log.Error("Unable to detect force push between: %s and %s in %-v Error: %v", oldCommitID, newCommitID, repo, err) | 		protectedFilePath := "" | ||||||
|  | 
 | ||||||
|  | 		globs := protectBranch.GetProtectedFilePatterns() | ||||||
|  | 		if len(globs) > 0 { | ||||||
|  | 			_, err := pull_service.CheckFileProtection(oldCommitID, newCommitID, globs, 1, env, gitRepo) | ||||||
|  | 			if err != nil { | ||||||
|  | 				if !models.IsErrFilePathProtected(err) { | ||||||
|  | 					log.Error("Unable to check file protection for commits from %s to %s in %-v: %v", oldCommitID, newCommitID, repo, err) | ||||||
| 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
| 						"err": fmt.Sprintf("Fail to detect force push: %v", err), | 						"err": fmt.Sprintf("Unable to check file protection for commits from %s to %s: %v", oldCommitID, newCommitID, err), | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} else if len(output) > 0 { |  | ||||||
| 					log.Warn("Forbidden: Branch: %s in %-v is protected from force push", branchName, repo) |  | ||||||
| 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("branch %s is protected from force push", branchName), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 
 |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			// Require signed commits
 |  | ||||||
| 			if protectBranch.RequireSignedCommits { |  | ||||||
| 				err := verifyCommits(oldCommitID, newCommitID, gitRepo, env) |  | ||||||
| 				if err != nil { |  | ||||||
| 					if !isErrUnverifiedCommit(err) { |  | ||||||
| 						log.Error("Unable to check commits from %s to %s in %-v: %v", oldCommitID, newCommitID, repo, err) |  | ||||||
| 						ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 							"err": fmt.Sprintf("Unable to check commits from %s to %s: %v", oldCommitID, newCommitID, err), |  | ||||||
| 						}) |  | ||||||
| 						return |  | ||||||
| 					} |  | ||||||
| 					unverifiedCommit := err.(*errUnverifiedCommit).sha |  | ||||||
| 					log.Warn("Forbidden: Branch: %s in %-v is protected from unverified commit %s", branchName, repo, unverifiedCommit) |  | ||||||
| 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("branch %s is protected from unverified commit %s", branchName, unverifiedCommit), |  | ||||||
| 					}) | 					}) | ||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 			} |  | ||||||
| 
 | 
 | ||||||
| 			// Detect Protected file pattern
 | 				changedProtectedfiles = true | ||||||
| 			globs := protectBranch.GetProtectedFilePatterns() | 				protectedFilePath = err.(models.ErrFilePathProtected).Path | ||||||
| 			if len(globs) > 0 { | 			} | ||||||
| 				err := checkFileProtection(oldCommitID, newCommitID, globs, gitRepo, env) | 		} | ||||||
| 				if err != nil { | 
 | ||||||
| 					if !models.IsErrFilePathProtected(err) { | 		// 5. Check if the doer is allowed to push
 | ||||||
| 						log.Error("Unable to check file protection for commits from %s to %s in %-v: %v", oldCommitID, newCommitID, repo, err) | 		canPush := false | ||||||
| 						ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | 		if opts.IsDeployKey { | ||||||
| 							"err": fmt.Sprintf("Unable to check file protection for commits from %s to %s: %v", oldCommitID, newCommitID, err), | 			canPush = !changedProtectedfiles && protectBranch.CanPush && (!protectBranch.EnableWhitelist || protectBranch.WhitelistDeployKeys) | ||||||
| 						}) | 		} else { | ||||||
| 						return | 			canPush = !changedProtectedfiles && protectBranch.CanUserPush(opts.UserID) | ||||||
| 					} | 		} | ||||||
| 					protectedFilePath := err.(models.ErrFilePathProtected).Path | 
 | ||||||
|  | 		// 6. If we're not allowed to push directly
 | ||||||
|  | 		if !canPush { | ||||||
|  | 			// Is this is a merge from the UI/API?
 | ||||||
|  | 			if opts.ProtectedBranchID == 0 { | ||||||
|  | 				// 6a. If we're not merging from the UI/API then there are two ways we got here:
 | ||||||
|  | 				//
 | ||||||
|  | 				// We are changing a protected file and we're not allowed to do that
 | ||||||
|  | 				if changedProtectedfiles { | ||||||
| 					log.Warn("Forbidden: Branch: %s in %-v is protected from changing file %s", branchName, repo, protectedFilePath) | 					log.Warn("Forbidden: Branch: %s in %-v is protected from changing file %s", branchName, repo, protectedFilePath) | ||||||
| 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ | 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
| 						"err": fmt.Sprintf("branch %s is protected from changing file %s", branchName, protectedFilePath), | 						"err": fmt.Sprintf("branch %s is protected from changing file %s", branchName, protectedFilePath), | ||||||
| 					}) | 					}) | ||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 			} |  | ||||||
| 
 | 
 | ||||||
| 			canPush := false | 				// Or we're simply not able to push to this protected branch
 | ||||||
| 			if opts.IsDeployKey { |  | ||||||
| 				canPush = protectBranch.CanPush && (!protectBranch.EnableWhitelist || protectBranch.WhitelistDeployKeys) |  | ||||||
| 			} else { |  | ||||||
| 				canPush = protectBranch.CanUserPush(opts.UserID) |  | ||||||
| 			} |  | ||||||
| 			if !canPush && opts.ProtectedBranchID > 0 { |  | ||||||
| 				// Merge (from UI or API)
 |  | ||||||
| 				pr, err := models.GetPullRequestByID(opts.ProtectedBranchID) |  | ||||||
| 				if err != nil { |  | ||||||
| 					log.Error("Unable to get PullRequest %d Error: %v", opts.ProtectedBranchID, err) |  | ||||||
| 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("Unable to get PullRequest %d Error: %v", opts.ProtectedBranchID, err), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				user, err := models.GetUserByID(opts.UserID) |  | ||||||
| 				if err != nil { |  | ||||||
| 					log.Error("Unable to get User id %d Error: %v", opts.UserID, err) |  | ||||||
| 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("Unable to get User id %d Error: %v", opts.UserID, err), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				perm, err := models.GetUserRepoPermission(repo, user) |  | ||||||
| 				if err != nil { |  | ||||||
| 					log.Error("Unable to get Repo permission of repo %s/%s of User %s", repo.OwnerName, repo.Name, user.Name, err) |  | ||||||
| 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("Unable to get Repo permission of repo %s/%s of User %s: %v", repo.OwnerName, repo.Name, user.Name, err), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				allowedMerge, err := pull_service.IsUserAllowedToMerge(pr, perm, user) |  | ||||||
| 				if err != nil { |  | ||||||
| 					log.Error("Error calculating if allowed to merge: %v", err) |  | ||||||
| 					ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("Error calculating if allowed to merge: %v", err), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				if !allowedMerge { |  | ||||||
| 					log.Warn("Forbidden: User %d is not allowed to push to protected branch: %s in %-v and is not allowed to merge pr #%d", opts.UserID, branchName, repo, pr.Index) |  | ||||||
| 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ |  | ||||||
| 						"err": fmt.Sprintf("Not allowed to push to protected branch %s", branchName), |  | ||||||
| 					}) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				// Check all status checks and reviews is ok, unless repo admin which can bypass this.
 |  | ||||||
| 				if !perm.IsAdmin() { |  | ||||||
| 					if err := pull_service.CheckPRReadyToMerge(pr); err != nil { |  | ||||||
| 						if models.IsErrNotAllowedToMerge(err) { |  | ||||||
| 							log.Warn("Forbidden: User %d is not allowed push to protected branch %s in %-v and pr #%d is not ready to be merged: %s", opts.UserID, branchName, repo, pr.Index, err.Error()) |  | ||||||
| 							ctx.JSON(http.StatusForbidden, map[string]interface{}{ |  | ||||||
| 								"err": fmt.Sprintf("Not allowed to push to protected branch %s and pr #%d is not ready to be merged: %s", branchName, opts.ProtectedBranchID, err.Error()), |  | ||||||
| 							}) |  | ||||||
| 							return |  | ||||||
| 						} |  | ||||||
| 						log.Error("Unable to check if mergable: protected branch %s in %-v and pr #%d. Error: %v", opts.UserID, branchName, repo, pr.Index, err) |  | ||||||
| 						ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ |  | ||||||
| 							"err": fmt.Sprintf("Unable to get status of pull request %d. Error: %v", opts.ProtectedBranchID, err), |  | ||||||
| 						}) |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			} else if !canPush { |  | ||||||
| 				log.Warn("Forbidden: User %d is not allowed to push to protected branch: %s in %-v", opts.UserID, branchName, repo) | 				log.Warn("Forbidden: User %d is not allowed to push to protected branch: %s in %-v", opts.UserID, branchName, repo) | ||||||
| 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
| 					"err": fmt.Sprintf("Not allowed to push to protected branch %s", branchName), | 					"err": fmt.Sprintf("Not allowed to push to protected branch %s", branchName), | ||||||
| 				}) | 				}) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
|  | 			// 6b. Merge (from UI or API)
 | ||||||
|  | 
 | ||||||
|  | 			// Get the PR, user and permissions for the user in the repository
 | ||||||
|  | 			pr, err := models.GetPullRequestByID(opts.ProtectedBranchID) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Error("Unable to get PullRequest %d Error: %v", opts.ProtectedBranchID, err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Unable to get PullRequest %d Error: %v", opts.ProtectedBranchID, err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			user, err := models.GetUserByID(opts.UserID) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Error("Unable to get User id %d Error: %v", opts.UserID, err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Unable to get User id %d Error: %v", opts.UserID, err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			perm, err := models.GetUserRepoPermission(repo, user) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Error("Unable to get Repo permission of repo %s/%s of User %s", repo.OwnerName, repo.Name, user.Name, err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Unable to get Repo permission of repo %s/%s of User %s: %v", repo.OwnerName, repo.Name, user.Name, err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			// Now check if the user is allowed to merge PRs for this repository
 | ||||||
|  | 			allowedMerge, err := pull_service.IsUserAllowedToMerge(pr, perm, user) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Error("Error calculating if allowed to merge: %v", err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Error calculating if allowed to merge: %v", err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if !allowedMerge { | ||||||
|  | 				log.Warn("Forbidden: User %d is not allowed to push to protected branch: %s in %-v and is not allowed to merge pr #%d", opts.UserID, branchName, repo, pr.Index) | ||||||
|  | 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Not allowed to push to protected branch %s", branchName), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			// If we're an admin for the repository we can ignore status checks, reviews and override protected files
 | ||||||
|  | 			if perm.IsAdmin() { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			// Now if we're not an admin - we can't overwrite protected files so fail now
 | ||||||
|  | 			if changedProtectedfiles { | ||||||
|  | 				log.Warn("Forbidden: Branch: %s in %-v is protected from changing file %s", branchName, repo, protectedFilePath) | ||||||
|  | 				ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("branch %s is protected from changing file %s", branchName, protectedFilePath), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			// Check all status checks and reviews are ok
 | ||||||
|  | 			if err := pull_service.CheckPRReadyToMerge(pr, true); err != nil { | ||||||
|  | 				if models.IsErrNotAllowedToMerge(err) { | ||||||
|  | 					log.Warn("Forbidden: User %d is not allowed push to protected branch %s in %-v and pr #%d is not ready to be merged: %s", opts.UserID, branchName, repo, pr.Index, err.Error()) | ||||||
|  | 					ctx.JSON(http.StatusForbidden, map[string]interface{}{ | ||||||
|  | 						"err": fmt.Sprintf("Not allowed to push to protected branch %s and pr #%d is not ready to be merged: %s", branchName, opts.ProtectedBranchID, err.Error()), | ||||||
|  | 					}) | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
|  | 				log.Error("Unable to check if mergable: protected branch %s in %-v and pr #%d. Error: %v", opts.UserID, branchName, repo, pr.Index, err) | ||||||
|  | 				ctx.JSON(http.StatusInternalServerError, map[string]interface{}{ | ||||||
|  | 					"err": fmt.Sprintf("Unable to get status of pull request %d. Error: %v", opts.ProtectedBranchID, err), | ||||||
|  | 				}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1426,6 +1426,9 @@ func ViewIssue(ctx *context.Context) { | ||||||
| 			ctx.Data["IsBlockedByOutdatedBranch"] = pull.ProtectedBranch.MergeBlockedByOutdatedBranch(pull) | 			ctx.Data["IsBlockedByOutdatedBranch"] = pull.ProtectedBranch.MergeBlockedByOutdatedBranch(pull) | ||||||
| 			ctx.Data["GrantedApprovals"] = cnt | 			ctx.Data["GrantedApprovals"] = cnt | ||||||
| 			ctx.Data["RequireSigned"] = pull.ProtectedBranch.RequireSignedCommits | 			ctx.Data["RequireSigned"] = pull.ProtectedBranch.RequireSignedCommits | ||||||
|  | 			ctx.Data["ChangedProtectedFiles"] = pull.ChangedProtectedFiles | ||||||
|  | 			ctx.Data["IsBlockedByChangedProtectedFiles"] = len(pull.ChangedProtectedFiles) != 0 | ||||||
|  | 			ctx.Data["ChangedProtectedFilesNum"] = len(pull.ChangedProtectedFiles) | ||||||
| 		} | 		} | ||||||
| 		ctx.Data["WillSign"] = false | 		ctx.Data["WillSign"] = false | ||||||
| 		if ctx.User != nil { | 		if ctx.User != nil { | ||||||
|  |  | ||||||
|  | @ -624,6 +624,20 @@ func ViewPullFiles(ctx *context.Context) { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err = pull.LoadProtectedBranch(); err != nil { | ||||||
|  | 		ctx.ServerError("LoadProtectedBranch", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if pull.ProtectedBranch != nil { | ||||||
|  | 		glob := pull.ProtectedBranch.GetProtectedFilePatterns() | ||||||
|  | 		if len(glob) != 0 { | ||||||
|  | 			for _, file := range diff.Files { | ||||||
|  | 				file.IsProtected = pull.ProtectedBranch.IsProtectedFile(glob, file.Name) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ctx.Data["Diff"] = diff | 	ctx.Data["Diff"] = diff | ||||||
| 	ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0 | 	ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0 | ||||||
| 
 | 
 | ||||||
|  | @ -772,7 +786,7 @@ func MergePullRequest(ctx *context.Context, form auth.MergePullRequestForm) { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := pull_service.CheckPRReadyToMerge(pr); err != nil { | 	if err := pull_service.CheckPRReadyToMerge(pr, false); err != nil { | ||||||
| 		if !models.IsErrNotAllowedToMerge(err) { | 		if !models.IsErrNotAllowedToMerge(err) { | ||||||
| 			ctx.ServerError("Merge PR status", err) | 			ctx.ServerError("Merge PR status", err) | ||||||
| 			return | 			return | ||||||
|  |  | ||||||
|  | @ -16,6 +16,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	pull_service "code.gitea.io/gitea/services/pull" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // ProtectedBranch render the page to protect the repository
 | // ProtectedBranch render the page to protect the repository
 | ||||||
|  | @ -262,6 +263,10 @@ func SettingsProtectedBranchPost(ctx *context.Context, f auth.ProtectBranchForm) | ||||||
| 			ctx.ServerError("UpdateProtectBranch", err) | 			ctx.ServerError("UpdateProtectBranch", err) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  | 		if err = pull_service.CheckPrsForBaseBranch(ctx.Repo.Repository, protectBranch.BranchName); err != nil { | ||||||
|  | 			ctx.ServerError("CheckPrsForBaseBranch", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		ctx.Flash.Success(ctx.Tr("repo.settings.update_protect_branch_success", branch)) | 		ctx.Flash.Success(ctx.Tr("repo.settings.update_protect_branch_success", branch)) | ||||||
| 		ctx.Redirect(fmt.Sprintf("%s/settings/branches/%s", ctx.Repo.RepoLink, branch)) | 		ctx.Redirect(fmt.Sprintf("%s/settings/branches/%s", ctx.Repo.RepoLink, branch)) | ||||||
| 	} else { | 	} else { | ||||||
|  |  | ||||||
|  | @ -351,6 +351,7 @@ type DiffFile struct { | ||||||
| 	IsSubmodule        bool | 	IsSubmodule        bool | ||||||
| 	Sections           []*DiffSection | 	Sections           []*DiffSection | ||||||
| 	IsIncomplete       bool | 	IsIncomplete       bool | ||||||
|  | 	IsProtected        bool | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetType returns type of diff file.
 | // GetType returns type of diff file.
 | ||||||
|  |  | ||||||
|  | @ -62,7 +62,7 @@ func checkAndUpdateStatus(pr *models.PullRequest) { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if !has { | 	if !has { | ||||||
| 		if err := pr.UpdateColsIfNotMerged("merge_base", "status", "conflicted_files"); err != nil { | 		if err := pr.UpdateColsIfNotMerged("merge_base", "status", "conflicted_files", "changed_protected_files"); err != nil { | ||||||
| 			log.Error("Update[%d]: %v", pr.ID, err) | 			log.Error("Update[%d]: %v", pr.ID, err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | @ -228,6 +228,20 @@ func handle(data ...queue.Data) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // CheckPrsForBaseBranch check all pulls with bseBrannch
 | ||||||
|  | func CheckPrsForBaseBranch(baseRepo *models.Repository, baseBranchName string) error { | ||||||
|  | 	prs, err := models.GetUnmergedPullRequestsByBaseInfo(baseRepo.ID, baseBranchName) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, pr := range prs { | ||||||
|  | 		AddToTaskQueue(pr) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Init runs the task queue to test all the checking status pull requests
 | // Init runs the task queue to test all the checking status pull requests
 | ||||||
| func Init() error { | func Init() error { | ||||||
| 	prQueue = queue.CreateUniqueQueue("pr_patch_checker", handle, "").(queue.UniqueQueue) | 	prQueue = queue.CreateUniqueQueue("pr_patch_checker", handle, "").(queue.UniqueQueue) | ||||||
|  |  | ||||||
|  | @ -559,7 +559,7 @@ func IsUserAllowedToMerge(pr *models.PullRequest, p models.Permission, user *mod | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // CheckPRReadyToMerge checks whether the PR is ready to be merged (reviews and status checks)
 | // CheckPRReadyToMerge checks whether the PR is ready to be merged (reviews and status checks)
 | ||||||
| func CheckPRReadyToMerge(pr *models.PullRequest) (err error) { | func CheckPRReadyToMerge(pr *models.PullRequest, skipProtectedFilesCheck bool) (err error) { | ||||||
| 	if err = pr.LoadBaseRepo(); err != nil { | 	if err = pr.LoadBaseRepo(); err != nil { | ||||||
| 		return fmt.Errorf("LoadBaseRepo: %v", err) | 		return fmt.Errorf("LoadBaseRepo: %v", err) | ||||||
| 	} | 	} | ||||||
|  | @ -598,5 +598,15 @@ func CheckPRReadyToMerge(pr *models.PullRequest) (err error) { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if skipProtectedFilesCheck { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if pr.ProtectedBranch.MergeBlockedByProtectedFiles(pr) { | ||||||
|  | 		return models.ErrNotAllowedToMerge{ | ||||||
|  | 			Reason: "Changed protected files", | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -18,6 +18,8 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/util" | 	"code.gitea.io/gitea/modules/util" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gobwas/glob" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // DownloadDiffOrPatch will write the patch for the pr to the writer
 | // DownloadDiffOrPatch will write the patch for the pr to the writer
 | ||||||
|  | @ -66,6 +68,7 @@ func TestPatch(pr *models.PullRequest) error { | ||||||
| 	} | 	} | ||||||
| 	defer gitRepo.Close() | 	defer gitRepo.Close() | ||||||
| 
 | 
 | ||||||
|  | 	// 1. update merge base
 | ||||||
| 	pr.MergeBase, err = git.NewCommand("merge-base", "--", "base", "tracking").RunInDir(tmpBasePath) | 	pr.MergeBase, err = git.NewCommand("merge-base", "--", "base", "tracking").RunInDir(tmpBasePath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		var err2 error | 		var err2 error | ||||||
|  | @ -75,10 +78,32 @@ func TestPatch(pr *models.PullRequest) error { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	pr.MergeBase = strings.TrimSpace(pr.MergeBase) | 	pr.MergeBase = strings.TrimSpace(pr.MergeBase) | ||||||
|  | 
 | ||||||
|  | 	// 2. Check for conflicts
 | ||||||
|  | 	if conflicts, err := checkConflicts(pr, gitRepo, tmpBasePath); err != nil || conflicts { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// 3. Check for protected files changes
 | ||||||
|  | 	if err = checkPullFilesProtection(pr, gitRepo); err != nil { | ||||||
|  | 		return fmt.Errorf("pr.CheckPullFilesProtection(): %v", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(pr.ChangedProtectedFiles) > 0 { | ||||||
|  | 		log.Trace("Found %d protected files changed", len(pr.ChangedProtectedFiles)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pr.Status = models.PullRequestStatusMergeable | ||||||
|  | 
 | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func checkConflicts(pr *models.PullRequest, gitRepo *git.Repository, tmpBasePath string) (bool, error) { | ||||||
|  | 	// 1. Create a plain patch from head to base
 | ||||||
| 	tmpPatchFile, err := ioutil.TempFile("", "patch") | 	tmpPatchFile, err := ioutil.TempFile("", "patch") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error("Unable to create temporary patch file! Error: %v", err) | 		log.Error("Unable to create temporary patch file! Error: %v", err) | ||||||
| 		return fmt.Errorf("Unable to create temporary patch file! Error: %v", err) | 		return false, fmt.Errorf("Unable to create temporary patch file! Error: %v", err) | ||||||
| 	} | 	} | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		_ = util.Remove(tmpPatchFile.Name()) | 		_ = util.Remove(tmpPatchFile.Name()) | ||||||
|  | @ -87,38 +112,43 @@ func TestPatch(pr *models.PullRequest) error { | ||||||
| 	if err := gitRepo.GetDiff(pr.MergeBase, "tracking", tmpPatchFile); err != nil { | 	if err := gitRepo.GetDiff(pr.MergeBase, "tracking", tmpPatchFile); err != nil { | ||||||
| 		tmpPatchFile.Close() | 		tmpPatchFile.Close() | ||||||
| 		log.Error("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) | 		log.Error("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) | ||||||
| 		return fmt.Errorf("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) | 		return false, fmt.Errorf("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) | ||||||
| 	} | 	} | ||||||
| 	stat, err := tmpPatchFile.Stat() | 	stat, err := tmpPatchFile.Stat() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		tmpPatchFile.Close() | 		tmpPatchFile.Close() | ||||||
| 		return fmt.Errorf("Unable to stat patch file: %v", err) | 		return false, fmt.Errorf("Unable to stat patch file: %v", err) | ||||||
| 	} | 	} | ||||||
| 	patchPath := tmpPatchFile.Name() | 	patchPath := tmpPatchFile.Name() | ||||||
| 	tmpPatchFile.Close() | 	tmpPatchFile.Close() | ||||||
| 
 | 
 | ||||||
|  | 	// 1a. if the size of that patch is 0 - there can be no conflicts!
 | ||||||
| 	if stat.Size() == 0 { | 	if stat.Size() == 0 { | ||||||
| 		log.Debug("PullRequest[%d]: Patch is empty - ignoring", pr.ID) | 		log.Debug("PullRequest[%d]: Patch is empty - ignoring", pr.ID) | ||||||
| 		pr.Status = models.PullRequestStatusMergeable | 		pr.Status = models.PullRequestStatusMergeable | ||||||
| 		pr.ConflictedFiles = []string{} | 		pr.ConflictedFiles = []string{} | ||||||
| 		return nil | 		return false, nil | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	log.Trace("PullRequest[%d].testPatch (patchPath): %s", pr.ID, patchPath) | 	log.Trace("PullRequest[%d].testPatch (patchPath): %s", pr.ID, patchPath) | ||||||
| 
 | 
 | ||||||
|  | 	// 2. preset the pr.Status as checking (this is not save at present)
 | ||||||
| 	pr.Status = models.PullRequestStatusChecking | 	pr.Status = models.PullRequestStatusChecking | ||||||
| 
 | 
 | ||||||
|  | 	// 3. Read the base branch in to the index of the temporary repository
 | ||||||
| 	_, err = git.NewCommand("read-tree", "base").RunInDir(tmpBasePath) | 	_, err = git.NewCommand("read-tree", "base").RunInDir(tmpBasePath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("git read-tree %s: %v", pr.BaseBranch, err) | 		return false, fmt.Errorf("git read-tree %s: %v", pr.BaseBranch, err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	// 4. Now get the pull request configuration to check if we need to ignore whitespace
 | ||||||
| 	prUnit, err := pr.BaseRepo.GetUnit(models.UnitTypePullRequests) | 	prUnit, err := pr.BaseRepo.GetUnit(models.UnitTypePullRequests) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return false, err | ||||||
| 	} | 	} | ||||||
| 	prConfig := prUnit.PullRequestsConfig() | 	prConfig := prUnit.PullRequestsConfig() | ||||||
| 
 | 
 | ||||||
|  | 	// 5. Prepare the arguments to apply the patch against the index
 | ||||||
| 	args := []string{"apply", "--check", "--cached"} | 	args := []string{"apply", "--check", "--cached"} | ||||||
| 	if prConfig.IgnoreWhitespaceConflicts { | 	if prConfig.IgnoreWhitespaceConflicts { | ||||||
| 		args = append(args, "--ignore-whitespace") | 		args = append(args, "--ignore-whitespace") | ||||||
|  | @ -126,26 +156,44 @@ func TestPatch(pr *models.PullRequest) error { | ||||||
| 	args = append(args, patchPath) | 	args = append(args, patchPath) | ||||||
| 	pr.ConflictedFiles = make([]string, 0, 5) | 	pr.ConflictedFiles = make([]string, 0, 5) | ||||||
| 
 | 
 | ||||||
|  | 	// 6. Prep the pipe:
 | ||||||
|  | 	//   - Here we could do the equivalent of:
 | ||||||
|  | 	//  `git apply --check --cached patch_file > conflicts`
 | ||||||
|  | 	//     Then iterate through the conflicts. However, that means storing all the conflicts
 | ||||||
|  | 	//     in memory - which is very wasteful.
 | ||||||
|  | 	//   - alternatively we can do the equivalent of:
 | ||||||
|  | 	//  `git apply --check ... | grep ...`
 | ||||||
|  | 	//     meaning we don't store all of the conflicts unnecessarily.
 | ||||||
| 	stderrReader, stderrWriter, err := os.Pipe() | 	stderrReader, stderrWriter, err := os.Pipe() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error("Unable to open stderr pipe: %v", err) | 		log.Error("Unable to open stderr pipe: %v", err) | ||||||
| 		return fmt.Errorf("Unable to open stderr pipe: %v", err) | 		return false, fmt.Errorf("Unable to open stderr pipe: %v", err) | ||||||
| 	} | 	} | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		_ = stderrReader.Close() | 		_ = stderrReader.Close() | ||||||
| 		_ = stderrWriter.Close() | 		_ = stderrWriter.Close() | ||||||
| 	}() | 	}() | ||||||
|  | 
 | ||||||
|  | 	// 7. Run the check command
 | ||||||
| 	conflict := false | 	conflict := false | ||||||
| 	err = git.NewCommand(args...). | 	err = git.NewCommand(args...). | ||||||
| 		RunInDirTimeoutEnvFullPipelineFunc( | 		RunInDirTimeoutEnvFullPipelineFunc( | ||||||
| 			nil, -1, tmpBasePath, | 			nil, -1, tmpBasePath, | ||||||
| 			nil, stderrWriter, nil, | 			nil, stderrWriter, nil, | ||||||
| 			func(ctx context.Context, cancel context.CancelFunc) error { | 			func(ctx context.Context, cancel context.CancelFunc) error { | ||||||
|  | 				// Close the writer end of the pipe to begin processing
 | ||||||
| 				_ = stderrWriter.Close() | 				_ = stderrWriter.Close() | ||||||
|  | 				defer func() { | ||||||
|  | 					// Close the reader on return to terminate the git command if necessary
 | ||||||
|  | 					_ = stderrReader.Close() | ||||||
|  | 				}() | ||||||
|  | 
 | ||||||
| 				const prefix = "error: patch failed:" | 				const prefix = "error: patch failed:" | ||||||
| 				const errorPrefix = "error: " | 				const errorPrefix = "error: " | ||||||
|  | 
 | ||||||
| 				conflictMap := map[string]bool{} | 				conflictMap := map[string]bool{} | ||||||
| 
 | 
 | ||||||
|  | 				// Now scan the output from the command
 | ||||||
| 				scanner := bufio.NewScanner(stderrReader) | 				scanner := bufio.NewScanner(stderrReader) | ||||||
| 				for scanner.Scan() { | 				for scanner.Scan() { | ||||||
| 					line := scanner.Text() | 					line := scanner.Text() | ||||||
|  | @ -170,25 +218,111 @@ func TestPatch(pr *models.PullRequest) error { | ||||||
| 						break | 						break | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
|  | 
 | ||||||
| 				if len(conflictMap) > 0 { | 				if len(conflictMap) > 0 { | ||||||
| 					pr.ConflictedFiles = make([]string, 0, len(conflictMap)) | 					pr.ConflictedFiles = make([]string, 0, len(conflictMap)) | ||||||
| 					for key := range conflictMap { | 					for key := range conflictMap { | ||||||
| 						pr.ConflictedFiles = append(pr.ConflictedFiles, key) | 						pr.ConflictedFiles = append(pr.ConflictedFiles, key) | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 				_ = stderrReader.Close() | 
 | ||||||
| 				return nil | 				return nil | ||||||
| 			}) | 			}) | ||||||
| 
 | 
 | ||||||
|  | 	// 8. If there is a conflict the `git apply` command will return a non-zero error code - so there will be a positive error.
 | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if conflict { | 		if conflict { | ||||||
| 			pr.Status = models.PullRequestStatusConflict | 			pr.Status = models.PullRequestStatusConflict | ||||||
| 			log.Trace("Found %d files conflicted: %v", len(pr.ConflictedFiles), pr.ConflictedFiles) | 			log.Trace("Found %d files conflicted: %v", len(pr.ConflictedFiles), pr.ConflictedFiles) | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		return fmt.Errorf("git apply --check: %v", err) |  | ||||||
| 	} |  | ||||||
| 	pr.Status = models.PullRequestStatusMergeable |  | ||||||
| 
 | 
 | ||||||
|  | 			return true, nil | ||||||
|  | 		} | ||||||
|  | 		return false, fmt.Errorf("git apply --check: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return false, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // CheckFileProtection check file Protection
 | ||||||
|  | func CheckFileProtection(oldCommitID, newCommitID string, patterns []glob.Glob, limit int, env []string, repo *git.Repository) ([]string, error) { | ||||||
|  | 	// 1. If there are no patterns short-circuit and just return nil
 | ||||||
|  | 	if len(patterns) == 0 { | ||||||
|  | 		return nil, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// 2. Prep the pipe
 | ||||||
|  | 	stdoutReader, stdoutWriter, err := os.Pipe() | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Error("Unable to create os.Pipe for %s", repo.Path) | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	defer func() { | ||||||
|  | 		_ = stdoutReader.Close() | ||||||
|  | 		_ = stdoutWriter.Close() | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	changedProtectedFiles := make([]string, 0, limit) | ||||||
|  | 
 | ||||||
|  | 	// 3. Run `git diff --name-only` to get the names of the changed files
 | ||||||
|  | 	err = git.NewCommand("diff", "--name-only", oldCommitID, newCommitID). | ||||||
|  | 		RunInDirTimeoutEnvFullPipelineFunc(env, -1, repo.Path, | ||||||
|  | 			stdoutWriter, nil, nil, | ||||||
|  | 			func(ctx context.Context, cancel context.CancelFunc) error { | ||||||
|  | 				// Close the writer end of the pipe to begin processing
 | ||||||
|  | 				_ = stdoutWriter.Close() | ||||||
|  | 				defer func() { | ||||||
|  | 					// Close the reader on return to terminate the git command if necessary
 | ||||||
|  | 					_ = stdoutReader.Close() | ||||||
|  | 				}() | ||||||
|  | 
 | ||||||
|  | 				// Now scan the output from the command
 | ||||||
|  | 				scanner := bufio.NewScanner(stdoutReader) | ||||||
|  | 				for scanner.Scan() { | ||||||
|  | 					path := strings.TrimSpace(scanner.Text()) | ||||||
|  | 					if len(path) == 0 { | ||||||
|  | 						continue | ||||||
|  | 					} | ||||||
|  | 					lpath := strings.ToLower(path) | ||||||
|  | 					for _, pat := range patterns { | ||||||
|  | 						if pat.Match(lpath) { | ||||||
|  | 							changedProtectedFiles = append(changedProtectedFiles, path) | ||||||
|  | 							break | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 					if len(changedProtectedFiles) >= limit { | ||||||
|  | 						break | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				if len(changedProtectedFiles) > 0 { | ||||||
|  | 					return models.ErrFilePathProtected{ | ||||||
|  | 						Path: changedProtectedFiles[0], | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				return scanner.Err() | ||||||
|  | 			}) | ||||||
|  | 	// 4. log real errors if there are any...
 | ||||||
|  | 	if err != nil && !models.IsErrFilePathProtected(err) { | ||||||
|  | 		log.Error("Unable to check file protection for commits from %s to %s in %s: %v", oldCommitID, newCommitID, repo.Path, err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return changedProtectedFiles, err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // checkPullFilesProtection check if pr changed protected files and save results
 | ||||||
|  | func checkPullFilesProtection(pr *models.PullRequest, gitRepo *git.Repository) error { | ||||||
|  | 	if err := pr.LoadProtectedBranch(); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if pr.ProtectedBranch == nil { | ||||||
|  | 		pr.ChangedProtectedFiles = nil | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var err error | ||||||
|  | 	pr.ChangedProtectedFiles, err = CheckFileProtection(pr.MergeBase, "tracking", pr.ProtectedBranch.GetProtectedFilePatterns(), 10, os.Environ(), gitRepo) | ||||||
|  | 	if err != nil && !models.IsErrFilePathProtected(err) { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -173,7 +173,7 @@ func ChangeTargetBranch(pr *models.PullRequest, doer *models.User, targetBranch | ||||||
| 	pr.CommitsAhead = divergence.Ahead | 	pr.CommitsAhead = divergence.Ahead | ||||||
| 	pr.CommitsBehind = divergence.Behind | 	pr.CommitsBehind = divergence.Behind | ||||||
| 
 | 
 | ||||||
| 	if err := pr.UpdateColsIfNotMerged("merge_base", "status", "conflicted_files", "base_branch", "commits_ahead", "commits_behind"); err != nil { | 	if err := pr.UpdateColsIfNotMerged("merge_base", "status", "conflicted_files", "changed_protected_files", "base_branch", "commits_ahead", "commits_behind"); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -68,6 +68,9 @@ | ||||||
| 						</div> | 						</div> | ||||||
| 						<span class="file">{{$file.Name}}</span> | 						<span class="file">{{$file.Name}}</span> | ||||||
| 						<div>{{$.i18n.Tr "repo.diff.file_suppressed"}}</div> | 						<div>{{$.i18n.Tr "repo.diff.file_suppressed"}}</div> | ||||||
|  | 						{{if $file.IsProtected}} | ||||||
|  | 							<span class="ui right basic label">{{$.i18n.Tr "repo.diff.protected"}}</span> | ||||||
|  | 						{{end}} | ||||||
| 						{{if and (not $file.IsSubmodule) (not $.PageIsWiki)}} | 						{{if and (not $file.IsSubmodule) (not $.PageIsWiki)}} | ||||||
| 							{{if $file.IsDeleted}} | 							{{if $file.IsDeleted}} | ||||||
| 								<a class="ui basic grey tiny button" rel="nofollow" href="{{EscapePound $.BeforeSourcePath}}/{{EscapePound .Name}}">{{$.i18n.Tr "repo.diff.view_file"}}</a> | 								<a class="ui basic grey tiny button" rel="nofollow" href="{{EscapePound $.BeforeSourcePath}}/{{EscapePound .Name}}">{{$.i18n.Tr "repo.diff.view_file"}}</a> | ||||||
|  | @ -104,6 +107,9 @@ | ||||||
| 							{{end}} | 							{{end}} | ||||||
| 						</div> | 						</div> | ||||||
| 						<span class="file">{{if $file.IsRenamed}}{{$file.OldName}} → {{end}}{{$file.Name}}{{if .IsLFSFile}} ({{$.i18n.Tr "repo.stored_lfs"}}){{end}}</span> | 						<span class="file">{{if $file.IsRenamed}}{{$file.OldName}} → {{end}}{{$file.Name}}{{if .IsLFSFile}} ({{$.i18n.Tr "repo.stored_lfs"}}){{end}}</span> | ||||||
|  | 						{{if $file.IsProtected}} | ||||||
|  | 							<span class="ui right basic label">{{$.i18n.Tr "repo.diff.protected"}}</span> | ||||||
|  | 						{{end}} | ||||||
| 						{{if and (not $file.IsSubmodule) (not $.PageIsWiki)}} | 						{{if and (not $file.IsSubmodule) (not $.PageIsWiki)}} | ||||||
| 							{{if $file.IsDeleted}} | 							{{if $file.IsDeleted}} | ||||||
| 								<a class="ui basic grey tiny button" rel="nofollow" href="{{EscapePound $.BeforeSourcePath}}/{{EscapePound .Name}}">{{$.i18n.Tr "repo.diff.view_file"}}</a> | 								<a class="ui basic grey tiny button" rel="nofollow" href="{{EscapePound $.BeforeSourcePath}}/{{EscapePound .Name}}">{{$.i18n.Tr "repo.diff.view_file"}}</a> | ||||||
|  |  | ||||||
|  | @ -67,6 +67,7 @@ | ||||||
| 	{{- else if .IsBlockedByApprovals}}red | 	{{- else if .IsBlockedByApprovals}}red | ||||||
| 	{{- else if .IsBlockedByRejection}}red | 	{{- else if .IsBlockedByRejection}}red | ||||||
| 	{{- else if .IsBlockedByOutdatedBranch}}red | 	{{- else if .IsBlockedByOutdatedBranch}}red | ||||||
|  | 	{{- else if .IsBlockedByChangedProtectedFiles}}red | ||||||
| 	{{- else if and .EnableStatusCheck (or .RequiredStatusCheckState.IsFailure .RequiredStatusCheckState.IsError)}}red | 	{{- else if and .EnableStatusCheck (or .RequiredStatusCheckState.IsFailure .RequiredStatusCheckState.IsError)}}red | ||||||
| 	{{- else if and .EnableStatusCheck (or (not $.LatestCommitStatus) .RequiredStatusCheckState.IsPending .RequiredStatusCheckState.IsWarning)}}yellow | 	{{- else if and .EnableStatusCheck (or (not $.LatestCommitStatus) .RequiredStatusCheckState.IsPending .RequiredStatusCheckState.IsWarning)}}yellow | ||||||
| 	{{- else if and .AllowMerge .RequireSigned (not .WillSign)}}red | 	{{- else if and .AllowMerge .RequireSigned (not .WillSign)}}red | ||||||
|  | @ -145,6 +146,16 @@ | ||||||
| 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | ||||||
| 					{{$.i18n.Tr "repo.pulls.blocked_by_outdated_branch"}} | 					{{$.i18n.Tr "repo.pulls.blocked_by_outdated_branch"}} | ||||||
| 					</div> | 					</div> | ||||||
|  | 				{{else if .IsBlockedByChangedProtectedFiles}} | ||||||
|  | 					<div class="item text red"> | ||||||
|  | 						<i class="icon icon-octicon">{{svg "octicon-x" 16}}</i> | ||||||
|  | 						{{$.i18n.Tr (TrN $.i18n.Lang $.ChangedProtectedFilesNum "repo.pulls.blocked_by_changed_protected_files_1" "repo.pulls.blocked_by_changed_protected_files_n") | Safe }} | ||||||
|  | 						<div class="ui ordered list"> | ||||||
|  | 							{{range .ChangedProtectedFiles}} | ||||||
|  | 								<div data-value="-" class="item">{{.}}</div> | ||||||
|  | 							{{end}} | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
| 				{{else if and .EnableStatusCheck (or .RequiredStatusCheckState.IsError .RequiredStatusCheckState.IsFailure)}} | 				{{else if and .EnableStatusCheck (or .RequiredStatusCheckState.IsError .RequiredStatusCheckState.IsFailure)}} | ||||||
| 					<div class="item text red"> | 					<div class="item text red"> | ||||||
| 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | ||||||
|  | @ -165,7 +176,7 @@ | ||||||
| 						{{$.i18n.Tr (printf "repo.signing.wont_sign.%s" .WontSignReason) }} | 						{{$.i18n.Tr (printf "repo.signing.wont_sign.%s" .WontSignReason) }} | ||||||
| 					</div> | 					</div> | ||||||
| 				{{end}} | 				{{end}} | ||||||
| 				{{$notAllOverridableChecksOk := or .IsBlockedByApprovals .IsBlockedByRejection .IsBlockedByOutdatedBranch (and .EnableStatusCheck (not .RequiredStatusCheckState.IsSuccess))}} | 				{{$notAllOverridableChecksOk := or .IsBlockedByApprovals .IsBlockedByRejection .IsBlockedByOutdatedBranch .IsBlockedByChangedProtectedFiles (and .EnableStatusCheck (not .RequiredStatusCheckState.IsSuccess))}} | ||||||
| 				{{if and (or $.IsRepoAdmin (not $notAllOverridableChecksOk)) (or (not .AllowMerge) (not .RequireSigned) .WillSign)}} | 				{{if and (or $.IsRepoAdmin (not $notAllOverridableChecksOk)) (or (not .AllowMerge) (not .RequireSigned) .WillSign)}} | ||||||
| 					{{if $notAllOverridableChecksOk}} | 					{{if $notAllOverridableChecksOk}} | ||||||
| 						<div class="item text yellow"> | 						<div class="item text yellow"> | ||||||
|  | @ -360,6 +371,16 @@ | ||||||
| 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | 						<i class="icon icon-octicon">{{svg "octicon-x"}}</i> | ||||||
| 					{{$.i18n.Tr "repo.pulls.blocked_by_outdated_branch"}} | 					{{$.i18n.Tr "repo.pulls.blocked_by_outdated_branch"}} | ||||||
| 					</div> | 					</div> | ||||||
|  | 				{{else if .IsBlockedByChangedProtectedFiles}} | ||||||
|  | 					<div class="item text red"> | ||||||
|  | 						<i class="icon icon-octicon">{{svg "octicon-x" 16}}</i> | ||||||
|  | 						{{$.i18n.Tr (TrN $.i18n.Lang $.ChangedProtectedFilesNum "repo.pulls.blocked_by_changed_protected_files_1" "repo.pulls.blocked_by_changed_protected_files_n") | Safe }} | ||||||
|  | 						<div class="ui ordered list"> | ||||||
|  | 							{{range .ChangedProtectedFiles}} | ||||||
|  | 								<div data-value="-" class="item">{{.}}</div> | ||||||
|  | 							{{end}} | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
| 				{{else if and .EnableStatusCheck (not .RequiredStatusCheckState.IsSuccess)}} | 				{{else if and .EnableStatusCheck (not .RequiredStatusCheckState.IsSuccess)}} | ||||||
| 					<div class="item text red"> | 					<div class="item text red"> | ||||||
| 						{{svg "octicon-x"}} | 						{{svg "octicon-x"}} | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue