2020-09-25 05:09:23 +01:00
// Copyright 2020 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repository
import (
2021-09-23 16:45:36 +01:00
"context"
2020-09-25 05:09:23 +01:00
"fmt"
"os"
"path/filepath"
"strings"
"code.gitea.io/gitea/models"
2021-09-19 19:49:59 +08:00
"code.gitea.io/gitea/models/db"
2021-12-10 09:27:50 +08:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-24 17:49:20 +08:00
user_model "code.gitea.io/gitea/models/user"
2020-09-25 05:09:23 +01:00
"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/log"
2021-11-16 21:30:11 +08:00
"code.gitea.io/gitea/modules/notification"
repo_module "code.gitea.io/gitea/modules/repository"
2020-09-25 05:09:23 +01:00
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/util"
2021-11-16 21:30:11 +08:00
2020-09-25 05:09:23 +01:00
"github.com/gobwas/glob"
)
2021-11-16 21:30:11 +08:00
// AdoptRepository adopts pre-existing repository files for the user/organization.
2021-12-10 09:27:50 +08:00
func AdoptRepository ( doer , u * user_model . User , opts models . CreateRepoOptions ) ( * repo_model . Repository , error ) {
2020-09-25 05:09:23 +01:00
if ! doer . IsAdmin && ! u . CanCreateRepo ( ) {
2021-12-12 23:48:20 +08:00
return nil , repo_model . ErrReachLimitOfRepo {
2020-09-25 05:09:23 +01:00
Limit : u . MaxRepoCreation ,
}
}
if len ( opts . DefaultBranch ) == 0 {
opts . DefaultBranch = setting . Repository . DefaultBranch
}
2021-12-10 09:27:50 +08:00
repo := & repo_model . Repository {
2020-09-25 05:09:23 +01:00
OwnerID : u . ID ,
Owner : u ,
OwnerName : u . Name ,
Name : opts . Name ,
LowerName : strings . ToLower ( opts . Name ) ,
Description : opts . Description ,
OriginalURL : opts . OriginalURL ,
OriginalServiceType : opts . GitServiceType ,
IsPrivate : opts . IsPrivate ,
IsFsckEnabled : ! opts . IsMirror ,
CloseIssuesViaCommitInAnyBranch : setting . Repository . DefaultCloseIssuesViaCommitsInAnyBranch ,
Status : opts . Status ,
IsEmpty : ! opts . AutoInit ,
}
2021-09-23 16:45:36 +01:00
if err := db . WithTx ( func ( ctx context . Context ) error {
2021-12-10 09:27:50 +08:00
repoPath := repo_model . RepoPath ( u . Name , repo . Name )
2020-11-28 02:42:08 +00:00
isExist , err := util . IsExist ( repoPath )
if err != nil {
log . Error ( "Unable to check if %s exists. Error: %v" , repoPath , err )
return err
}
if ! isExist {
2021-12-10 09:27:50 +08:00
return repo_model . ErrRepoNotExist {
2020-09-25 05:09:23 +01:00
OwnerName : u . Name ,
Name : repo . Name ,
}
}
if err := models . CreateRepository ( ctx , doer , u , repo , true ) ; err != nil {
return err
}
if err := adoptRepository ( ctx , repoPath , doer , repo , opts ) ; err != nil {
return fmt . Errorf ( "createDelegateHooks: %v" , err )
}
2021-12-10 09:27:50 +08:00
if err := models . CheckDaemonExportOK ( ctx , repo ) ; err != nil {
2021-10-13 20:47:02 +01:00
return fmt . Errorf ( "checkDaemonExportOK: %v" , err )
}
2020-09-25 05:09:23 +01:00
// Initialize Issue Labels if selected
if len ( opts . IssueLabels ) > 0 {
if err := models . InitializeLabels ( ctx , repo . ID , opts . IssueLabels , false ) ; err != nil {
return fmt . Errorf ( "InitializeLabels: %v" , err )
}
}
2022-02-06 20:01:47 +01:00
if stdout , err := git . NewCommand ( ctx , "update-server-info" ) .
2020-09-25 05:09:23 +01:00
SetDescription ( fmt . Sprintf ( "CreateRepository(git update-server-info): %s" , repoPath ) ) .
RunInDir ( repoPath ) ; err != nil {
log . Error ( "CreateRepository(git update-server-info) in %v: Stdout: %s\nError: %v" , repo , stdout , err )
return fmt . Errorf ( "CreateRepository(git update-server-info): %v" , err )
}
return nil
} ) ; err != nil {
return nil , err
}
2021-11-16 21:30:11 +08:00
notification . NotifyCreateRepository ( doer , u , repo )
2020-09-25 05:09:23 +01:00
return repo , nil
}
2021-12-10 09:27:50 +08:00
func adoptRepository ( ctx context . Context , repoPath string , u * user_model . User , repo * repo_model . Repository , opts models . CreateRepoOptions ) ( err error ) {
2021-11-16 21:30:11 +08:00
isExist , err := util . IsExist ( repoPath )
if err != nil {
log . Error ( "Unable to check if %s exists. Error: %v" , repoPath , err )
return err
}
if ! isExist {
return fmt . Errorf ( "adoptRepository: path does not already exist: %s" , repoPath )
}
if err := repo_module . CreateDelegateHooks ( repoPath ) ; err != nil {
return fmt . Errorf ( "createDelegateHooks: %v" , err )
}
// Re-fetch the repository from database before updating it (else it would
// override changes that were done earlier with sql)
2021-12-10 09:27:50 +08:00
if repo , err = repo_model . GetRepositoryByIDCtx ( ctx , repo . ID ) ; err != nil {
2021-11-16 21:30:11 +08:00
return fmt . Errorf ( "getRepositoryByID: %v" , err )
}
repo . IsEmpty = false
2022-01-19 23:26:57 +00:00
// Don't bother looking this repo in the context it won't be there
gitRepo , err := git . OpenRepositoryCtx ( ctx , repo . RepoPath ( ) )
2021-11-16 21:30:11 +08:00
if err != nil {
return fmt . Errorf ( "openRepository: %v" , err )
}
defer gitRepo . Close ( )
2022-01-19 23:26:57 +00:00
2021-11-16 21:30:11 +08:00
if len ( opts . DefaultBranch ) > 0 {
repo . DefaultBranch = opts . DefaultBranch
if err = gitRepo . SetDefaultBranch ( repo . DefaultBranch ) ; err != nil {
return fmt . Errorf ( "setDefaultBranch: %v" , err )
}
} else {
repo . DefaultBranch , err = gitRepo . GetDefaultBranch ( )
if err != nil {
repo . DefaultBranch = setting . Repository . DefaultBranch
if err = gitRepo . SetDefaultBranch ( repo . DefaultBranch ) ; err != nil {
return fmt . Errorf ( "setDefaultBranch: %v" , err )
}
}
repo . DefaultBranch = strings . TrimPrefix ( repo . DefaultBranch , git . BranchPrefix )
}
2021-12-08 19:08:16 +00:00
branches , _ , _ := gitRepo . GetBranchNames ( 0 , 0 )
2021-11-16 21:30:11 +08:00
found := false
hasDefault := false
hasMaster := false
hasMain := false
for _ , branch := range branches {
if branch == repo . DefaultBranch {
found = true
break
} else if branch == setting . Repository . DefaultBranch {
hasDefault = true
} else if branch == "master" {
hasMaster = true
} else if branch == "main" {
hasMain = true
}
}
if ! found {
if hasDefault {
repo . DefaultBranch = setting . Repository . DefaultBranch
} else if hasMaster {
repo . DefaultBranch = "master"
} else if hasMain {
repo . DefaultBranch = "main"
} else if len ( branches ) > 0 {
repo . DefaultBranch = branches [ 0 ]
} else {
repo . IsEmpty = true
repo . DefaultBranch = setting . Repository . DefaultBranch
}
if err = gitRepo . SetDefaultBranch ( repo . DefaultBranch ) ; err != nil {
return fmt . Errorf ( "setDefaultBranch: %v" , err )
}
}
if err = models . UpdateRepositoryCtx ( ctx , repo , false ) ; err != nil {
return fmt . Errorf ( "updateRepository: %v" , err )
}
return nil
}
2020-09-25 05:09:23 +01:00
// DeleteUnadoptedRepository deletes unadopted repository files from the filesystem
2021-11-24 17:49:20 +08:00
func DeleteUnadoptedRepository ( doer , u * user_model . User , repoName string ) error {
2021-12-12 23:48:20 +08:00
if err := repo_model . IsUsableRepoName ( repoName ) ; err != nil {
2020-09-25 05:09:23 +01:00
return err
}
2021-12-10 09:27:50 +08:00
repoPath := repo_model . RepoPath ( u . Name , repoName )
2020-11-28 02:42:08 +00:00
isExist , err := util . IsExist ( repoPath )
if err != nil {
log . Error ( "Unable to check if %s exists. Error: %v" , repoPath , err )
return err
}
if ! isExist {
2021-12-10 09:27:50 +08:00
return repo_model . ErrRepoNotExist {
2020-09-25 05:09:23 +01:00
OwnerName : u . Name ,
Name : repoName ,
}
}
2021-12-10 09:27:50 +08:00
if exist , err := repo_model . IsRepositoryExist ( u , repoName ) ; err != nil {
2020-09-25 05:09:23 +01:00
return err
} else if exist {
2021-12-12 23:48:20 +08:00
return repo_model . ErrRepoAlreadyExist {
2020-09-25 05:09:23 +01:00
Uname : u . Name ,
Name : repoName ,
}
}
return util . RemoveAll ( repoPath )
}
2022-01-01 03:52:00 +01:00
type unadoptedRrepositories struct {
repositories [ ] string
index int
start int
end int
}
func ( unadopted * unadoptedRrepositories ) add ( repository string ) {
if unadopted . index >= unadopted . start && unadopted . index < unadopted . end {
unadopted . repositories = append ( unadopted . repositories , repository )
}
unadopted . index ++
}
func checkUnadoptedRepositories ( userName string , repoNamesToCheck [ ] string , unadopted * unadoptedRrepositories ) error {
if len ( repoNamesToCheck ) == 0 {
return nil
}
ctxUser , err := user_model . GetUserByName ( userName )
if err != nil {
if user_model . IsErrUserNotExist ( err ) {
log . Debug ( "Missing user: %s" , userName )
return nil
}
return err
}
repos , _ , err := models . GetUserRepositories ( & models . SearchRepoOptions {
Actor : ctxUser ,
Private : true ,
ListOptions : db . ListOptions {
Page : 1 ,
PageSize : len ( repoNamesToCheck ) ,
2022-01-20 18:46:10 +01:00
} , LowerNames : repoNamesToCheck ,
} )
2022-01-01 03:52:00 +01:00
if err != nil {
return err
}
if len ( repos ) == len ( repoNamesToCheck ) {
return nil
}
repoNames := make ( map [ string ] bool , len ( repos ) )
for _ , repo := range repos {
repoNames [ repo . LowerName ] = true
}
for _ , repoName := range repoNamesToCheck {
if _ , ok := repoNames [ repoName ] ; ! ok {
unadopted . add ( filepath . Join ( userName , repoName ) )
}
}
return nil
}
2020-09-25 05:09:23 +01:00
// ListUnadoptedRepositories lists all the unadopted repositories that match the provided query
2021-09-24 19:32:56 +08:00
func ListUnadoptedRepositories ( query string , opts * db . ListOptions ) ( [ ] string , int , error ) {
2020-09-25 05:09:23 +01:00
globUser , _ := glob . Compile ( "*" )
globRepo , _ := glob . Compile ( "*" )
qsplit := strings . SplitN ( query , "/" , 2 )
if len ( qsplit ) > 0 && len ( query ) > 0 {
var err error
globUser , err = glob . Compile ( qsplit [ 0 ] )
if err != nil {
2021-07-08 07:38:13 -04:00
log . Info ( "Invalid glob expression '%s' (skipped): %v" , qsplit [ 0 ] , err )
2020-09-25 05:09:23 +01:00
}
if len ( qsplit ) > 1 {
globRepo , err = glob . Compile ( qsplit [ 1 ] )
if err != nil {
2021-07-08 07:38:13 -04:00
log . Info ( "Invalid glob expression '%s' (skipped): %v" , qsplit [ 1 ] , err )
2020-09-25 05:09:23 +01:00
}
}
}
2022-01-01 03:52:00 +01:00
var repoNamesToCheck [ ] string
2020-09-25 05:09:23 +01:00
2022-01-01 03:52:00 +01:00
start := ( opts . Page - 1 ) * opts . PageSize
unadopted := & unadoptedRrepositories {
repositories : make ( [ ] string , 0 , opts . PageSize ) ,
start : start ,
end : start + opts . PageSize ,
index : 0 ,
}
2020-09-25 05:09:23 +01:00
2022-01-01 03:52:00 +01:00
var userName string
2020-09-25 05:09:23 +01:00
// We're going to iterate by pagesize.
2021-11-15 06:02:53 +00:00
root := filepath . Clean ( setting . RepoRootPath )
2020-09-25 05:09:23 +01:00
if err := filepath . Walk ( root , func ( path string , info os . FileInfo , err error ) error {
if err != nil {
return err
}
if ! info . IsDir ( ) || path == root {
return nil
}
if ! strings . ContainsRune ( path [ len ( root ) + 1 : ] , filepath . Separator ) {
// Got a new user
2022-01-01 03:52:00 +01:00
if err = checkUnadoptedRepositories ( userName , repoNamesToCheck , unadopted ) ; err != nil {
return err
2020-09-25 05:09:23 +01:00
}
2022-01-01 03:52:00 +01:00
repoNamesToCheck = repoNamesToCheck [ : 0 ]
2020-09-25 05:09:23 +01:00
if ! globUser . Match ( info . Name ( ) ) {
return filepath . SkipDir
}
2022-01-01 03:52:00 +01:00
userName = info . Name ( )
2020-09-25 05:09:23 +01:00
return nil
}
name := info . Name ( )
if ! strings . HasSuffix ( name , ".git" ) {
return filepath . SkipDir
}
name = name [ : len ( name ) - 4 ]
2021-12-12 23:48:20 +08:00
if repo_model . IsUsableRepoName ( name ) != nil || strings . ToLower ( name ) != name || ! globRepo . Match ( name ) {
2020-09-25 05:09:23 +01:00
return filepath . SkipDir
}
2022-01-01 03:52:00 +01:00
repoNamesToCheck = append ( repoNamesToCheck , name )
2020-09-25 05:09:23 +01:00
return filepath . SkipDir
} ) ; err != nil {
return nil , 0 , err
}
2022-01-01 03:52:00 +01:00
if err := checkUnadoptedRepositories ( userName , repoNamesToCheck , unadopted ) ; err != nil {
return nil , 0 , err
2020-09-25 05:09:23 +01:00
}
2022-01-01 03:52:00 +01:00
return unadopted . repositories , unadopted . index , nil
2020-09-25 05:09:23 +01:00
}