2019-06-01 18:00:21 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
// Package private includes all internal routes. The package name internal is ideal but Golang is not allowed, so we use private as package name instead.
package private
import (
"fmt"
"net/http"
"strings"
2021-12-10 11:14:24 +03:00
asymkey_model "code.gitea.io/gitea/models/asymkey"
2021-11-28 14:58:28 +03:00
"code.gitea.io/gitea/models/perm"
2022-05-11 13:09:36 +03:00
access_model "code.gitea.io/gitea/models/perm/access"
2021-12-10 04:27:50 +03:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-09 22:57:58 +03:00
"code.gitea.io/gitea/models/unit"
2021-11-24 12:49:20 +03:00
user_model "code.gitea.io/gitea/models/user"
2021-01-26 18:36:53 +03:00
"code.gitea.io/gitea/modules/context"
2021-07-28 12:42:56 +03:00
"code.gitea.io/gitea/modules/git"
2019-06-01 18:00:21 +03:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/private"
"code.gitea.io/gitea/modules/setting"
2019-12-15 05:49:52 +03:00
repo_service "code.gitea.io/gitea/services/repository"
2020-01-07 21:27:36 +03:00
wiki_service "code.gitea.io/gitea/services/wiki"
2019-06-01 18:00:21 +03:00
)
// ServNoCommand returns information about the provided keyid
2021-01-26 18:36:53 +03:00
func ServNoCommand ( ctx * context . PrivateContext ) {
2019-06-01 18:00:21 +03:00
keyID := ctx . ParamsInt64 ( ":keyid" )
if keyID <= 0 {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusBadRequest , private . Response {
Err : fmt . Sprintf ( "Bad key id: %d" , keyID ) ,
2019-06-01 18:00:21 +03:00
} )
}
results := private . KeyAndOwner { }
2021-12-10 11:14:24 +03:00
key , err := asymkey_model . GetPublicKeyByID ( keyID )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-12-10 11:14:24 +03:00
if asymkey_model . IsErrKeyNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . Response {
Err : fmt . Sprintf ( "Cannot find key: %d" , keyID ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
log . Error ( "Unable to get public key: %d Error: %v" , keyID , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . Response {
Err : err . Error ( ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
results . Key = key
2021-12-10 11:14:24 +03:00
if key . Type == asymkey_model . KeyTypeUser || key . Type == asymkey_model . KeyTypePrincipal {
2021-11-24 12:49:20 +03:00
user , err := user_model . GetUserByID ( key . OwnerID )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-11-24 12:49:20 +03:00
if user_model . IsErrUserNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . Response {
Err : fmt . Sprintf ( "Cannot find owner with id: %d for key: %d" , key . OwnerID , keyID ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
log . Error ( "Unable to get owner with id: %d for public key: %d Error: %v" , key . OwnerID , keyID , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . Response {
Err : err . Error ( ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
2020-11-13 02:29:11 +03:00
if ! user . IsActive || user . ProhibitLogin {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . Response {
Err : "Your account is disabled." ,
2020-11-13 02:29:11 +03:00
} )
return
}
2019-06-01 18:00:21 +03:00
results . Owner = user
}
ctx . JSON ( http . StatusOK , & results )
}
// ServCommand returns information about the provided keyid
2021-01-26 18:36:53 +03:00
func ServCommand ( ctx * context . PrivateContext ) {
2019-06-01 18:00:21 +03:00
keyID := ctx . ParamsInt64 ( ":keyid" )
ownerName := ctx . Params ( ":owner" )
repoName := ctx . Params ( ":repo" )
2021-11-28 14:58:28 +03:00
mode := perm . AccessMode ( ctx . FormInt ( "mode" ) )
2019-06-01 18:00:21 +03:00
// Set the basic parts of the results to return
results := private . ServCommandResults {
RepoName : repoName ,
OwnerName : ownerName ,
KeyID : keyID ,
}
2020-08-20 17:53:06 +03:00
// Now because we're not translating things properly let's just default some English strings here
2019-06-01 18:00:21 +03:00
modeString := "read"
2021-11-28 14:58:28 +03:00
if mode > perm . AccessModeRead {
2019-06-01 18:00:21 +03:00
modeString = "write to"
}
// The default unit we're trying to look at is code
2021-11-09 22:57:58 +03:00
unitType := unit . TypeCode
2019-06-01 18:00:21 +03:00
// Unless we're a wiki...
if strings . HasSuffix ( repoName , ".wiki" ) {
// in which case we need to look at the wiki
2021-11-09 22:57:58 +03:00
unitType = unit . TypeWiki
2019-06-01 18:00:21 +03:00
// And we'd better munge the reponame and tell downstream we're looking at a wiki
results . IsWiki = true
results . RepoName = repoName [ : len ( repoName ) - 5 ]
}
2021-11-24 12:49:20 +03:00
owner , err := user_model . GetUserByName ( results . OwnerName )
2020-11-13 02:29:11 +03:00
if err != nil {
2022-01-16 06:44:11 +03:00
if user_model . IsErrUserNotExist ( err ) {
// User is fetching/cloning a non-existent repository
log . Warn ( "Failed authentication attempt (cannot find repository: %s/%s) from %s" , results . OwnerName , results . RepoName , ctx . RemoteAddr ( ) )
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Cannot find repository: %s/%s" , results . OwnerName , results . RepoName ) ,
} )
return
}
2020-11-13 02:29:11 +03:00
log . Error ( "Unable to get repository owner: %s/%s Error: %v" , results . OwnerName , results . RepoName , err )
2022-01-16 06:44:11 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
2021-06-23 22:38:19 +03:00
Results : results ,
Err : fmt . Sprintf ( "Unable to get repository owner: %s/%s %v" , results . OwnerName , results . RepoName , err ) ,
2020-11-13 02:29:11 +03:00
} )
return
}
2020-11-18 12:58:25 +03:00
if ! owner . IsOrganization ( ) && ! owner . IsActive {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
Results : results ,
Err : "Repository cannot be accessed, you could retry it later" ,
2020-11-13 02:29:11 +03:00
} )
return
}
2019-06-01 18:00:21 +03:00
// Now get the Repository and set the results section
2019-12-15 05:49:52 +03:00
repoExist := true
2021-12-10 04:27:50 +03:00
repo , err := repo_model . GetRepositoryByName ( owner . ID , results . RepoName )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-12-10 04:27:50 +03:00
if repo_model . IsErrRepoNotExist ( err ) {
2019-12-15 05:49:52 +03:00
repoExist = false
2021-07-29 04:42:15 +03:00
for _ , verb := range ctx . FormStrings ( "verb" ) {
2020-02-05 12:40:35 +03:00
if "git-upload-pack" == verb {
// User is fetching/cloning a non-existent repository
2022-01-16 06:44:11 +03:00
log . Warn ( "Failed authentication attempt (cannot find repository: %s/%s) from %s" , results . OwnerName , results . RepoName , ctx . RemoteAddr ( ) )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Cannot find repository: %s/%s" , results . OwnerName , results . RepoName ) ,
2020-02-05 12:40:35 +03:00
} )
return
}
}
2019-12-15 05:49:52 +03:00
} else {
log . Error ( "Unable to get repository: %s/%s Error: %v" , results . OwnerName , results . RepoName , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get repository: %s/%s %v" , results . OwnerName , results . RepoName , err ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
}
2019-12-15 05:49:52 +03:00
if repoExist {
2020-11-13 02:29:11 +03:00
repo . Owner = owner
2019-12-15 05:49:52 +03:00
repo . OwnerName = ownerName
results . RepoID = repo . ID
2019-10-13 16:23:14 +03:00
2019-12-15 05:49:52 +03:00
if repo . IsBeingCreated ( ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : "Repository is being created, you could retry after it finished" ,
2019-12-15 05:49:52 +03:00
} )
return
}
2021-11-23 01:32:16 +03:00
if repo . IsBroken ( ) {
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : "Repository is in a broken state" ,
} )
return
}
2019-12-15 05:49:52 +03:00
// We can shortcut at this point if the repo is a mirror
2021-11-28 14:58:28 +03:00
if mode > perm . AccessModeRead && repo . IsMirror {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Mirror Repository %s/%s is read-only" , results . OwnerName , results . RepoName ) ,
2019-12-15 05:49:52 +03:00
} )
return
}
2019-06-01 18:00:21 +03:00
}
// Get the Public Key represented by the keyID
2021-12-10 11:14:24 +03:00
key , err := asymkey_model . GetPublicKeyByID ( keyID )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-12-10 11:14:24 +03:00
if asymkey_model . IsErrKeyNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Cannot find key: %d" , keyID ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
log . Error ( "Unable to get public key: %d Error: %v" , keyID , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get key: %d Error: %v" , keyID , err ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
results . KeyName = key . Name
results . KeyID = key . ID
results . UserID = key . OwnerID
2019-12-15 05:49:52 +03:00
// If repo doesn't exist, deploy key doesn't make sense
2021-12-10 11:14:24 +03:00
if ! repoExist && key . Type == asymkey_model . KeyTypeDeploy {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Cannot find repository %s/%s" , results . OwnerName , results . RepoName ) ,
2019-12-15 05:49:52 +03:00
} )
return
}
2019-06-01 18:00:21 +03:00
// Deploy Keys have ownerID set to 0 therefore we can't use the owner
// So now we need to check if the key is a deploy key
// We'll keep hold of the deploy key here for permissions checking
2021-12-10 11:14:24 +03:00
var deployKey * asymkey_model . DeployKey
2021-11-24 12:49:20 +03:00
var user * user_model . User
2021-12-10 11:14:24 +03:00
if key . Type == asymkey_model . KeyTypeDeploy {
2019-06-01 18:00:21 +03:00
var err error
2021-12-10 11:14:24 +03:00
deployKey , err = asymkey_model . GetDeployKeyByRepo ( key . ID , repo . ID )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-12-10 11:14:24 +03:00
if asymkey_model . IsErrDeployKeyNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Public (Deploy) Key: %d:%s is not authorized to %s %s/%s." , key . ID , key . Name , modeString , results . OwnerName , results . RepoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
log . Error ( "Unable to get deploy for public (deploy) key: %d in %-v Error: %v" , key . ID , repo , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get Deploy Key for Public Key: %d:%s in %s/%s." , key . ID , key . Name , results . OwnerName , results . RepoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
2022-03-22 12:29:07 +03:00
results . DeployKeyID = deployKey . ID
2019-06-01 18:00:21 +03:00
results . KeyName = deployKey . Name
// FIXME: Deploy keys aren't really the owner of the repo pushing changes
// however we don't have good way of representing deploy keys in hook.go
// so for now use the owner of the repository
results . UserName = results . OwnerName
results . UserID = repo . OwnerID
2020-08-30 10:24:39 +03:00
if ! repo . Owner . KeepEmailPrivate {
results . UserEmail = repo . Owner . Email
}
2019-06-01 18:00:21 +03:00
} else {
// Get the user represented by the Key
var err error
2021-11-24 12:49:20 +03:00
user , err = user_model . GetUserByID ( key . OwnerID )
2019-06-01 18:00:21 +03:00
if err != nil {
2021-11-24 12:49:20 +03:00
if user_model . IsErrUserNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Public Key: %d:%s owner %d does not exist." , key . ID , key . Name , key . OwnerID ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
log . Error ( "Unable to get owner: %d for public key: %d:%s Error: %v" , key . OwnerID , key . ID , key . Name , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get Owner: %d for Deploy Key: %d:%s in %s/%s." , key . OwnerID , key . ID , key . Name , ownerName , repoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
2020-11-13 02:29:11 +03:00
if ! user . IsActive || user . ProhibitLogin {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . Response {
Err : "Your account is disabled." ,
2020-11-13 02:29:11 +03:00
} )
return
}
2019-06-01 18:00:21 +03:00
results . UserName = user . Name
2020-08-30 10:24:39 +03:00
if ! user . KeepEmailPrivate {
results . UserEmail = user . Email
}
2019-06-01 18:00:21 +03:00
}
// Don't allow pushing if the repo is archived
2021-11-28 14:58:28 +03:00
if repoExist && mode > perm . AccessModeRead && repo . IsArchived {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Repo: %s/%s is archived." , results . OwnerName , results . RepoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
// Permissions checking:
2021-10-20 23:59:05 +03:00
if repoExist &&
2021-11-28 14:58:28 +03:00
( mode > perm . AccessModeRead ||
2021-10-20 23:59:05 +03:00
repo . IsPrivate ||
owner . Visibility . IsPrivate ( ) ||
2021-10-26 02:24:19 +03:00
( user != nil && user . IsRestricted ) || // user will be nil if the key is a deploykey
2021-10-20 23:59:05 +03:00
setting . Service . RequireSignInView ) {
2021-12-10 11:14:24 +03:00
if key . Type == asymkey_model . KeyTypeDeploy {
2019-06-01 18:00:21 +03:00
if deployKey . Mode < mode {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Deploy Key: %d:%s is not authorized to %s %s/%s." , key . ID , key . Name , modeString , results . OwnerName , results . RepoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
} else {
2021-10-20 23:59:05 +03:00
// Because of the special ref "refs/for" we will need to delay write permission check
2021-11-09 22:57:58 +03:00
if git . SupportProcReceive && unitType == unit . TypeCode {
2021-11-28 14:58:28 +03:00
mode = perm . AccessModeRead
2021-07-28 12:42:56 +03:00
}
2022-05-11 13:09:36 +03:00
perm , err := access_model . GetUserRepoPermission ( ctx , repo , user )
2019-06-01 18:00:21 +03:00
if err != nil {
log . Error ( "Unable to get permissions for %-v with key %d in %-v Error: %v" , user , key . ID , repo , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get permissions for user %d:%s with key %d in %s/%s Error: %v" , user . ID , user . Name , key . ID , results . OwnerName , results . RepoName , err ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
userMode := perm . UnitAccessMode ( unitType )
if userMode < mode {
2022-01-16 06:44:11 +03:00
log . Warn ( "Failed authentication attempt for %s with key %s (not authorized to %s %s/%s) from %s" , user . Name , key . Name , modeString , ownerName , repoName , ctx . RemoteAddr ( ) )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusUnauthorized , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "User: %d:%s with Key: %d:%s is not authorized to %s %s/%s." , user . ID , user . Name , key . ID , key . Name , modeString , ownerName , repoName ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
}
}
2019-12-15 05:49:52 +03:00
// We already know we aren't using a deploy key
if ! repoExist {
2021-11-24 12:49:20 +03:00
owner , err := user_model . GetUserByName ( ownerName )
2019-12-15 05:49:52 +03:00
if err != nil {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Unable to get owner: %s %v" , results . OwnerName , err ) ,
2019-12-15 05:49:52 +03:00
} )
return
}
if owner . IsOrganization ( ) && ! setting . Repository . EnablePushCreateOrg {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
Results : results ,
Err : "Push to create is not enabled for organizations." ,
2019-12-15 05:49:52 +03:00
} )
return
}
if ! owner . IsOrganization ( ) && ! setting . Repository . EnablePushCreateUser {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
Results : results ,
Err : "Push to create is not enabled for users." ,
2019-12-15 05:49:52 +03:00
} )
return
}
repo , err = repo_service . PushCreateRepo ( user , owner , results . RepoName )
if err != nil {
log . Error ( "pushCreateRepo: %v" , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusNotFound , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Cannot find repository: %s/%s" , results . OwnerName , results . RepoName ) ,
2019-12-15 05:49:52 +03:00
} )
return
}
results . RepoID = repo . ID
}
2019-06-01 18:00:21 +03:00
if results . IsWiki {
2020-04-19 17:26:58 +03:00
// Ensure the wiki is enabled before we allow access to it
2021-11-09 22:57:58 +03:00
if _ , err := repo . GetUnit ( unit . TypeWiki ) ; err != nil {
2021-12-10 04:27:50 +03:00
if repo_model . IsErrUnitTypeNotExist ( err ) {
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusForbidden , private . ErrServCommand {
Results : results ,
Err : "repository wiki is disabled" ,
2020-04-19 17:26:58 +03:00
} )
return
}
log . Error ( "Failed to get the wiki unit in %-v Error: %v" , repo , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Failed to get the wiki unit in %s/%s Error: %v" , ownerName , repoName , err ) ,
2020-04-19 17:26:58 +03:00
} )
return
}
// Finally if we're trying to touch the wiki we should init it
2022-01-20 02:26:57 +03:00
if err = wiki_service . InitWiki ( ctx , repo ) ; err != nil {
2019-06-01 18:00:21 +03:00
log . Error ( "Failed to initialize the wiki in %-v Error: %v" , repo , err )
2021-06-23 22:38:19 +03:00
ctx . JSON ( http . StatusInternalServerError , private . ErrServCommand {
Results : results ,
Err : fmt . Sprintf ( "Failed to initialize the wiki in %s/%s Error: %v" , ownerName , repoName , err ) ,
2019-06-01 18:00:21 +03:00
} )
return
}
}
2022-03-22 12:29:07 +03:00
log . Debug ( "Serv Results:\nIsWiki: %t\nDeployKeyID: %d\nKeyID: %d\tKeyName: %s\nUserName: %s\nUserID: %d\nOwnerName: %s\nRepoName: %s\nRepoID: %d" ,
2019-06-01 18:00:21 +03:00
results . IsWiki ,
2022-03-22 12:29:07 +03:00
results . DeployKeyID ,
2019-06-01 18:00:21 +03:00
results . KeyID ,
results . KeyName ,
results . UserName ,
results . UserID ,
results . OwnerName ,
results . RepoName ,
results . RepoID )
ctx . JSON ( http . StatusOK , results )
// We will update the keys in a different call.
}