2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								// Copyright 2019 The Gitea Authors. All rights reserved.  
						 
					
						
							
								
									
										
										
										
											2022-11-27 13:20:29 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// SPDX-License-Identifier: MIT  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								package  private  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  (  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"fmt" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"net/http" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"os" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"code.gitea.io/gitea/models" 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									asymkey_model  "code.gitea.io/gitea/models/asymkey" 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-12 23:51:54 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									git_model  "code.gitea.io/gitea/models/git" 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-13 17:37:59 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									issues_model  "code.gitea.io/gitea/models/issues" 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									perm_model  "code.gitea.io/gitea/models/perm" 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-11 18:09:36 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									access_model  "code.gitea.io/gitea/models/perm/access" 
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 03:57:58 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									"code.gitea.io/gitea/models/unit" 
							 
						 
					
						
							
								
									
										
										
										
											2021-11-24 17:49:20 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									user_model  "code.gitea.io/gitea/models/user" 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									gitea_context  "code.gitea.io/gitea/modules/context" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"code.gitea.io/gitea/modules/git" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"code.gitea.io/gitea/modules/log" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"code.gitea.io/gitea/modules/private" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"code.gitea.io/gitea/modules/web" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									pull_service  "code.gitea.io/gitea/services/pull" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								)  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								type  preReceiveContext  struct  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									* gitea_context . PrivateContext 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// loadedPusher indicates that where the following information are loaded 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									loadedPusher         bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									user                 * user_model . User  // it's the org user if a DeployKey is used 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-11 18:09:36 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									userPerm             access_model . Permission 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									deployKeyAccessMode  perm_model . AccessMode 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									canCreatePullRequest         bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									checkedCanCreatePullRequest  bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									canWriteCode         bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									checkedCanWriteCode  bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-12 23:51:54 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									protectedTags     [ ] * git_model . ProtectedTag 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									gotProtectedTags  bool 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									env  [ ] string 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									opts  * private . HookOptions 
							 
						 
					
						
							
								
									
										
										
										
											2022-04-28 17:45:33 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									branchName  string 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// CanWriteCode returns true if pusher can write code  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								func  ( ctx  * preReceiveContext )  CanWriteCode ( )  bool  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! ctx . checkedCanWriteCode  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ! ctx . loadPusherAndPermission ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-22 22:14:27 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . canWriteCode  =  issues_model . CanMaintainerWriteToBranch ( ctx ,  ctx . userPerm ,  ctx . branchName ,  ctx . user )  ||  ctx . deployKeyAccessMode  >=  perm_model . AccessModeWrite 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										ctx . checkedCanWriteCode  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  ctx . canWriteCode 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// AssertCanWriteCode returns true if pusher can write code  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								func  ( ctx  * preReceiveContext )  AssertCanWriteCode ( )  bool  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! ctx . CanWriteCode ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ctx . Written ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											UserMsg :  "User permission denied for writing." , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// CanCreatePullRequest returns true if pusher can create pull requests  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								func  ( ctx  * preReceiveContext )  CanCreatePullRequest ( )  bool  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! ctx . checkedCanCreatePullRequest  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ! ctx . loadPusherAndPermission ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . canCreatePullRequest  =  ctx . userPerm . CanRead ( unit . TypePullRequests ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										ctx . checkedCanCreatePullRequest  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  ctx . canCreatePullRequest 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// AssertCreatePullRequest returns true if can create pull requests  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								func  ( ctx  * preReceiveContext )  AssertCreatePullRequest ( )  bool  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! ctx . CanCreatePullRequest ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ctx . Written ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											UserMsg :  "User permission denied for creating pull-request." , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// HookPreReceive checks whether a individual commit is acceptable  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  HookPreReceive ( ctx  * gitea_context . PrivateContext )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									opts  :=  web . GetForm ( ctx ) . ( * private . HookOptions ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									ourCtx  :=  & preReceiveContext { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										PrivateContext :  ctx , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										env :             generateGitEnv ( opts ) ,  // Generate git environment for checking commits 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										opts :            opts , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// Iterate across the provided old commit IDs 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  i  :=  range  opts . OldCommitIDs  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										oldCommitID  :=  opts . OldCommitIDs [ i ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										newCommitID  :=  opts . NewCommitIDs [ i ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										refFullName  :=  opts . RefFullNames [ i ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										switch  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										case  refFullName . IsBranch ( ) : 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											preReceiveBranch ( ourCtx ,  oldCommitID ,  newCommitID ,  refFullName ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										case  refFullName . IsTag ( ) : 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											preReceiveTag ( ourCtx ,  oldCommitID ,  newCommitID ,  refFullName ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										case  git . SupportProcReceive  &&  refFullName . IsFor ( ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											preReceiveFor ( ourCtx ,  oldCommitID ,  newCommitID ,  refFullName ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										default : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ourCtx . AssertCanWriteCode ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ctx . Written ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-15 14:59:57 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									ctx . PlainText ( http . StatusOK ,  "ok" ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								func  preReceiveBranch ( ctx  * preReceiveContext ,  oldCommitID ,  newCommitID  string ,  refFullName  git . RefName )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									branchName  :=  refFullName . BranchName ( ) 
							 
						 
					
						
							
								
									
										
										
										
											2022-04-28 17:45:33 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									ctx . branchName  =  branchName 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ! ctx . AssertCanWriteCode ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									repo  :=  ctx . Repo . Repository 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									gitRepo  :=  ctx . Repo . GitRepo 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 21:02:00 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									objectFormat ,  _  :=  gitRepo . GetObjectFormat ( ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-17 19:56:08 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  branchName  ==  repo . DefaultBranch  &&  newCommitID  ==  objectFormat . EmptyObjectID ( ) . String ( )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										log . Warn ( "Forbidden: Branch: %s is the default branch in %-v and cannot be deleted" ,  branchName ,  repo ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											UserMsg :  fmt . Sprintf ( "branch %s is the default branch and cannot be deleted" ,  branchName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-16 16:00:22 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									protectBranch ,  err  :=  git_model . GetFirstMatchProtectedBranchRule ( ctx ,  repo . ID ,  branchName ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										log . Error ( "Unable to get protected branch: %s in %-v Error: %v" ,  branchName ,  repo ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											Err :  err . Error ( ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// Allow pushes to non-protected branches 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-16 16:00:22 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  protectBranch  ==  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-16 16:00:22 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									protectBranch . Repo  =  repo 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// This ref is a protected branch. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// First of all we need to enforce absolutely: 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 1. Detect and prevent deletion of the branch 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-17 19:56:08 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  newCommitID  ==  objectFormat . EmptyObjectID ( ) . String ( )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										log . Warn ( "Forbidden: Branch: %s in %-v is protected from deletion" ,  branchName ,  repo ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											UserMsg :  fmt . Sprintf ( "branch %s is protected from deletion" ,  branchName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 2. Disallow force pushes to protected branches 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-17 19:56:08 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  oldCommitID  !=  objectFormat . EmptyObjectID ( ) . String ( )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-23 22:44:45 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										output ,  _ ,  err  :=  git . NewCommand ( ctx ,  "rev-list" ,  "--max-count=1" ) . AddDynamicArguments ( oldCommitID ,  "^" + newCommitID ) . RunStdString ( & git . RunOpts { Dir :  repo . RepoPath ( ) ,  Env :  ctx . env } ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										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 ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												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 ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												UserMsg :  fmt . Sprintf ( "branch %s is protected from force push" ,  branchName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 3. Enforce require signed commits 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  protectBranch . RequireSignedCommits  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										err  :=  verifyCommits ( oldCommitID ,  newCommitID ,  gitRepo ,  ctx . 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 ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													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 ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												UserMsg :  fmt . Sprintf ( "branch %s is protected from unverified commit %s" ,  branchName ,  unverifiedCommit ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// Now there are several tests which can be overridden: 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 4. Check protected file patterns - this is overridable from the UI 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									changedProtectedfiles  :=  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									protectedFilePath  :=  "" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									globs  :=  protectBranch . GetProtectedFilePatterns ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  len ( globs )  >  0  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 23:26:57 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										_ ,  err  :=  pull_service . CheckFileProtection ( gitRepo ,  oldCommitID ,  newCommitID ,  globs ,  1 ,  ctx . env ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										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 ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													Err :  fmt . Sprintf ( "Unable to check file protection for commits from %s to %s: %v" ,  oldCommitID ,  newCommitID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											changedProtectedfiles  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											protectedFilePath  =  err . ( models . ErrFilePathProtected ) . Path 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 5. Check if the doer is allowed to push 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-20 12:02:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									var  canPush  bool 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ctx . opts . DeployKeyID  !=  0  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										canPush  =  ! changedProtectedfiles  &&  protectBranch . CanPush  &&  ( ! protectBranch . EnableWhitelist  ||  protectBranch . WhitelistDeployKeys ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									}  else  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-16 16:00:22 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										user ,  err  :=  user_model . GetUserByID ( ctx ,  ctx . opts . UserID ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to GetUserByID for commits from %s to %s in %-v: %v" ,  oldCommitID ,  newCommitID ,  repo ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to GetUserByID for commits from %s to %s: %v" ,  oldCommitID ,  newCommitID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										canPush  =  ! changedProtectedfiles  &&  protectBranch . CanUserPush ( ctx ,  user ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// 6. If we're not allowed to push directly 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! canPush  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Is this is a merge from the UI/API? 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ctx . opts . PullRequestID  ==  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 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													UserMsg :  fmt . Sprintf ( "branch %s is protected from changing file %s" ,  branchName ,  protectedFilePath ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											// Allow commits that only touch unprotected files 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											globs  :=  protectBranch . GetUnprotectedFilePatterns ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  len ( globs )  >  0  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 23:26:57 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												unprotectedFilesOnly ,  err  :=  pull_service . CheckUnprotectedFiles ( gitRepo ,  oldCommitID ,  newCommitID ,  globs ,  ctx . env ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													log . Error ( "Unable to check file protection for commits from %s to %s in %-v: %v" ,  oldCommitID ,  newCommitID ,  repo ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
														Err :  fmt . Sprintf ( "Unable to check file protection for commits from %s to %s: %v" ,  oldCommitID ,  newCommitID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												if  unprotectedFilesOnly  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													// Commit only touches unprotected files, this is allowed 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											// Or we're simply not able to push to this protected branch 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Warn ( "Forbidden: User %d is not allowed to push to protected branch: %s in %-v" ,  ctx . opts . UserID ,  branchName ,  repo ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												UserMsg :  fmt . Sprintf ( "Not allowed to push to protected branch %s" ,  branchName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// 6b. Merge (from UI or API) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Get the PR, user and permissions for the user in the repository 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-13 17:37:59 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										pr ,  err  :=  issues_model . GetPullRequestByID ( ctx ,  ctx . opts . PullRequestID ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get PullRequest %d Error: %v" ,  ctx . opts . PullRequestID ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get PullRequest %d Error: %v" ,  ctx . opts . PullRequestID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// although we should have called `loadPusherAndPermission` before, here we call it explicitly again because we need to access ctx.user below 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ! ctx . loadPusherAndPermission ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											// if error occurs, loadPusherAndPermission had written the error response 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										// Now check if the user is allowed to merge PRs for this repository 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Note: we can use ctx.perm and ctx.user directly as they will have been loaded above 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-03 21:46:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										allowedMerge ,  err  :=  pull_service . IsUserAllowedToMerge ( ctx ,  pr ,  ctx . userPerm ,  ctx . user ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Error calculating if allowed to merge: %v" ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												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" ,  ctx . opts . UserID ,  branchName ,  repo ,  pr . Index ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												UserMsg :  fmt . Sprintf ( "Not allowed to push to protected branch %s" ,  branchName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// If we're an admin for the repository we can ignore status checks, reviews and override protected files 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ctx . userPerm . IsAdmin ( )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// 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 ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												UserMsg :  fmt . Sprintf ( "branch %s is protected from changing file %s" ,  branchName ,  protectedFilePath ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Check all status checks and reviews are ok 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-02 01:54:44 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  err  :=  pull_service . CheckPullBranchProtections ( ctx ,  pr ,  true ) ;  err  !=  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-31 16:53:08 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  models . IsErrDisallowedToMerge ( err )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												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" ,  ctx . opts . UserID ,  branchName ,  repo ,  pr . Index ,  err . Error ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													UserMsg :  fmt . Sprintf ( "Not allowed to push to protected branch %s and pr #%d is not ready to be merged: %s" ,  branchName ,  ctx . opts . PullRequestID ,  err . Error ( ) ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to check if mergable: protected branch %s in %-v and pr #%d. Error: %v" ,  ctx . opts . UserID ,  branchName ,  repo ,  pr . Index ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get status of pull request %d. Error: %v" ,  ctx . opts . PullRequestID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								func  preReceiveTag ( ctx  * preReceiveContext ,  oldCommitID ,  newCommitID  string ,  refFullName  git . RefName )  {  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ! ctx . AssertCanWriteCode ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									tagName  :=  refFullName . TagName ( ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! ctx . gotProtectedTags  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										var  err  error 
							 
						 
					
						
							
								
									
										
										
										
											2022-11-19 09:12:33 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . protectedTags ,  err  =  git_model . GetProtectedTags ( ctx ,  ctx . Repo . Repository . ID ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get protected tags for %-v Error: %v" ,  ctx . Repo . Repository ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  err . Error ( ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . gotProtectedTags  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-11-19 09:12:33 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									isAllowed ,  err  :=  git_model . IsUserAllowedToControlTag ( ctx ,  ctx . protectedTags ,  tagName ,  ctx . opts . UserID ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											Err :  err . Error ( ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! isAllowed  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										log . Warn ( "Forbidden: Tag %s in %-v is protected" ,  tagName ,  ctx . Repo . Repository ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											UserMsg :  fmt . Sprintf ( "Tag %s is protected" ,  tagName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								func  preReceiveFor ( ctx  * preReceiveContext ,  oldCommitID ,  newCommitID  string ,  refFullName  git . RefName )  {  
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ! ctx . AssertCreatePullRequest ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ctx . Repo . Repository . IsEmpty  { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											UserMsg :  "Can't create pull request for an empty repository." , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ctx . opts . IsWiki  { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											UserMsg :  "Pull requests are not supported on the wiki." , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-26 09:04:48 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									baseBranchName  :=  refFullName . ForBranchName ( ) 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									baseBranchExist  :=  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ctx . Repo . GitRepo . IsBranchExist ( baseBranchName )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										baseBranchExist  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! baseBranchExist  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										for  p ,  v  :=  range  baseBranchName  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  v  ==  '/'  &&  ctx . Repo . GitRepo . IsBranchExist ( baseBranchName [ : p ] )  &&  p  !=  len ( baseBranchName ) - 1  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												baseBranchExist  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												break 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ! baseBranchExist  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . JSON ( http . StatusForbidden ,  private . Response { 
							 
						 
					
						
							
								
									
										
											 
										
											
												Refactor internal API for git commands, use meaningful messages instead of "Internal Server Error" (#23687)
# Why this PR comes
At first, I'd like to help users like #23636 (there are a lot)
The unclear "Internal Server Error" is quite anonying, scare users,
frustrate contributors, nobody knows what happens.
So, it's always good to provide meaningful messages to end users (of
course, do not leak sensitive information).
When I started working on the "response message to end users", I found
that the related code has a lot of technical debt. A lot of copy&paste
code, unclear fields and usages.
So I think it's good to make everything clear.
# Tech Backgrounds
Gitea has many sub-commands, some are used by admins, some are used by
SSH servers or Git Hooks. Many sub-commands use "internal API" to
communicate with Gitea web server.
Before, Gitea server always use `StatusCode + Json "err" field` to
return messages.
* The CLI sub-commands: they expect to show all error related messages
to site admin
* The Serv/Hook sub-commands (for git clients): they could only show
safe messages to end users, the error log could only be recorded by
"SSHLog" to Gitea web server.
In the old design, it assumes that:
* If the StatusCode is 500 (in some functions), then the "err" field is
error log, shouldn't be exposed to git client.
* If the StatusCode is 40x, then the "err" field could be exposed. And
some functions always read the "err" no matter what the StatusCode is.
The old code is not strict, and it's difficult to distinguish the
messages clearly and then output them correctly.
# This PR
To help to remove duplicate code and make everything clear, this PR
introduces `ResponseExtra` and `requestJSONResp`.
* `ResponseExtra` is a struct which contains "extra" information of a
internal API response, including StatusCode, UserMsg, Error
* `requestJSONResp` is a generic function which can be used for all
cases to help to simplify the calls.
* Remove all `map["err"]`, always use `private.Response{Err}` to
construct error messages.
* User messages and error messages are separated clearly, the `fail` and
`handleCliResponseExtra` will output correct messages.
* Replace all `Internal Server Error` messages with meaningful (still
safe) messages.
This PR saves more than 300 lines, while makes the git client messages
more clear.
Many gitea-serv/git-hook related essential functions are covered by
tests.
---------
Co-authored-by: delvh <dev.lh@web.de>
											 
										 
										
											2023-03-29 14:32:26 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											UserMsg :  fmt . Sprintf ( "Unexpected ref: %s" ,  refFullName ) , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  generateGitEnv ( opts  * private . HookOptions )  ( env  [ ] string )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									env  =  os . Environ ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  opts . GitAlternativeObjectDirectories  !=  ""  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										env  =  append ( env , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											private . GitAlternativeObjectDirectories + "=" + opts . GitAlternativeObjectDirectories ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  opts . GitObjectDirectory  !=  ""  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										env  =  append ( env , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											private . GitObjectDirectory + "=" + opts . GitObjectDirectory ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  opts . GitQuarantinePath  !=  ""  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										env  =  append ( env , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											private . GitQuarantinePath + "=" + opts . GitQuarantinePath ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  env 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// loadPusherAndPermission returns false if an error occurs, and it writes the error response  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  ( ctx  * preReceiveContext )  loadPusherAndPermission ( )  bool  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ctx . loadedPusher  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
											 
										
											
												Implement actions (#21937)
Close #13539.
Co-authored by: @lunny @appleboy @fuxiaohei and others.
Related projects:
- https://gitea.com/gitea/actions-proto-def
- https://gitea.com/gitea/actions-proto-go
- https://gitea.com/gitea/act
- https://gitea.com/gitea/act_runner
### Summary
The target of this PR is to bring a basic implementation of "Actions",
an internal CI/CD system of Gitea. That means even though it has been
merged, the state of the feature is **EXPERIMENTAL**, and please note
that:
- It is disabled by default;
- It shouldn't be used in a production environment currently;
- It shouldn't be used in a public Gitea instance currently;
- Breaking changes may be made before it's stable.
**Please comment on #13539 if you have any different product design
ideas**, all decisions reached there will be adopted here. But in this
PR, we don't talk about **naming, feature-creep or alternatives**.
### ⚠️ Breaking
`gitea-actions` will become a reserved user name. If a user with the
name already exists in the database, it is recommended to rename it.
### Some important reviews
- What is `DEFAULT_ACTIONS_URL` in `app.ini` for?
  - https://github.com/go-gitea/gitea/pull/21937#discussion_r1055954954
- Why the api for runners is not under the normal `/api/v1` prefix?
  - https://github.com/go-gitea/gitea/pull/21937#discussion_r1061173592
- Why DBFS?
  - https://github.com/go-gitea/gitea/pull/21937#discussion_r1061301178
- Why ignore events triggered by `gitea-actions` bot?
  - https://github.com/go-gitea/gitea/pull/21937#discussion_r1063254103
- Why there's no permission control for actions?
  - https://github.com/go-gitea/gitea/pull/21937#discussion_r1090229868
### What it looks like
<details>
#### Manage runners
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205870657-c72f590e-2e08-4cd4-be7f-2e0abb299bbf.png">
#### List runs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872794-50fde990-2b45-48c1-a178-908e4ec5b627.png">
#### View logs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872501-9b7b9000-9542-4991-8f55-18ccdada77c3.png">
</details>
### How to try it
<details>
#### 1. Start Gitea
Clone this branch and [install from
source](https://docs.gitea.io/en-us/install-from-source).
Add additional configurations in `app.ini` to enable Actions:
```ini
[actions]
ENABLED = true
```
Start it.
If all is well, you'll see the management page of runners:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205877365-8e30a780-9b10-4154-b3e8-ee6c3cb35a59.png">
#### 2. Start runner
Clone the [act_runner](https://gitea.com/gitea/act_runner), and follow
the
[README](https://gitea.com/gitea/act_runner/src/branch/main/README.md)
to start it.
If all is well, you'll see a new runner has been added:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205878000-216f5937-e696-470d-b66c-8473987d91c3.png">
#### 3. Enable actions for a repo
Create a new repo or open an existing one, check the `Actions` checkbox
in settings and submit.
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879705-53e09208-73c0-4b3e-a123-2dcf9aba4b9c.png">
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879383-23f3d08f-1a85-41dd-a8b3-54e2ee6453e8.png">
If all is well, you'll see a new tab "Actions":
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205881648-a8072d8c-5803-4d76-b8a8-9b2fb49516c1.png">
#### 4. Upload workflow files
Upload some workflow files to `.gitea/workflows/xxx.yaml`, you can
follow the [quickstart](https://docs.github.com/en/actions/quickstart)
of GitHub Actions. Yes, Gitea Actions is compatible with GitHub Actions
in most cases, you can use the same demo:
```yaml
name: GitHub Actions Demo
run-name: ${{ github.actor }} is testing out GitHub Actions 🚀
on: [push]
jobs:
  Explore-GitHub-Actions:
    runs-on: ubuntu-latest
    steps:
      - run: echo "🎉 The job was automatically triggered by a ${{ github.event_name }} event."
      - run: echo "🐧 This job is now running on a ${{ runner.os }} server hosted by GitHub!"
      - run: echo "🔎 The name of your branch is ${{ github.ref }} and your repository is ${{ github.repository }}."
      - name: Check out repository code
        uses: actions/checkout@v3
      - run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner."
      - run: echo "🖥️ The workflow is now ready to test your code on the runner."
      - name: List files in the repository
        run: |
          ls ${{ github.workspace }}
      - run: echo "🍏 This job's status is ${{ job.status }}."
```
If all is well, you'll see a new run in `Actions` tab:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884473-79a874bc-171b-4aaf-acd5-0241a45c3b53.png">
#### 5. Check the logs of jobs
Click a run and you'll see the logs:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884800-994b0374-67f7-48ff-be9a-4c53f3141547.png">
#### 6. Go on
You can try more examples in [the
documents](https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions)
of GitHub Actions, then you might find a lot of bugs.
Come on, PRs are welcome.
</details>
See also: [Feature Preview: Gitea
Actions](https://blog.gitea.io/2022/12/feature-preview-gitea-actions/)
---------
Co-authored-by: a1012112796 <1012112796@qq.com>
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
Co-authored-by: delvh <dev.lh@web.de>
Co-authored-by: ChristopherHX <christopher.homberger@web.de>
Co-authored-by: John Olheiser <john.olheiser@gmail.com>
											 
										 
										
											2023-01-31 09:45:19 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ctx . opts . UserID  ==  user_model . ActionsUserID  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . user  =  user_model . NewActionsUser ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . userPerm . AccessMode  =  perm_model . AccessMode ( ctx . opts . ActionPerm ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  :=  ctx . Repo . Repository . LoadUnits ( ctx ) ;  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get User id %d Error: %v" ,  ctx . opts . UserID ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get User id %d Error: %v" ,  ctx . opts . UserID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . userPerm . Units  =  ctx . Repo . Repository . Units 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . userPerm . UnitsMode  =  make ( map [ unit . Type ] perm_model . AccessMode ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										for  _ ,  u  :=  range  ctx . Repo . Repository . Units  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . userPerm . UnitsMode [ u . Type ]  =  ctx . userPerm . AccessMode 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										user ,  err  :=  user_model . GetUserByID ( ctx ,  ctx . opts . UserID ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get User id %d Error: %v" ,  ctx . opts . UserID ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get User id %d Error: %v" ,  ctx . opts . UserID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . user  =  user 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										userPerm ,  err  :=  access_model . GetUserRepoPermission ( ctx ,  ctx . Repo . Repository ,  user ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get Repo permission of repo %s/%s of User %s: %v" ,  ctx . Repo . Repository . OwnerName ,  ctx . Repo . Repository . Name ,  user . Name ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get Repo permission of repo %s/%s of User %s: %v" ,  ctx . Repo . Repository . OwnerName ,  ctx . Repo . Repository . Name ,  user . Name ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . userPerm  =  userPerm 
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ctx . opts . DeployKeyID  !=  0  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										deployKey ,  err  :=  asymkey_model . GetDeployKeyByID ( ctx ,  ctx . opts . DeployKeyID ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											log . Error ( "Unable to get DeployKey id %d Error: %v" ,  ctx . opts . DeployKeyID ,  err ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											ctx . JSON ( http . StatusInternalServerError ,  private . Response { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												Err :  fmt . Sprintf ( "Unable to get DeployKey id %d Error: %v" ,  ctx . opts . DeployKeyID ,  err ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  false 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ctx . deployKeyAccessMode  =  deployKey . Mode 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-03-22 17:29:07 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									ctx . loadedPusher  =  true 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  true 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-16 14:34:54 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}