2019-12-14 20:30:01 +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 repository
import (
2020-12-02 21:36:06 +03:00
"context"
2019-12-14 20:30:01 +03:00
"fmt"
2021-04-09 01:25:57 +03:00
"net/url"
2019-12-14 20:30:01 +03:00
"path"
"strings"
"time"
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/git"
2021-04-09 01:25:57 +03:00
"code.gitea.io/gitea/modules/lfs"
2019-12-14 20:30:01 +03:00
"code.gitea.io/gitea/modules/log"
2020-09-11 01:29:19 +03:00
migration "code.gitea.io/gitea/modules/migrations/base"
2019-12-14 20:30:01 +03:00
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/timeutil"
2020-08-11 23:05:34 +03:00
"code.gitea.io/gitea/modules/util"
2020-01-20 23:01:19 +03:00
2019-12-14 20:30:01 +03:00
"gopkg.in/ini.v1"
)
/ *
GitHub , GitLab , Gogs : * . wiki . git
BitBucket : * . git / wiki
* /
var commonWikiURLSuffixes = [ ] string { ".wiki.git" , ".git/wiki" }
2020-07-06 05:08:32 +03:00
// WikiRemoteURL returns accessible repository URL for wiki if exists.
2019-12-14 20:30:01 +03:00
// Otherwise, it returns an empty string.
2020-07-06 05:08:32 +03:00
func WikiRemoteURL ( remote string ) string {
2019-12-14 20:30:01 +03:00
remote = strings . TrimSuffix ( remote , ".git" )
for _ , suffix := range commonWikiURLSuffixes {
wikiURL := remote + suffix
if git . IsRepoURLAccessible ( wikiURL ) {
return wikiURL
}
}
return ""
}
// MigrateRepositoryGitData starts migrating git related data after created migrating repository
2020-12-02 21:36:06 +03:00
func MigrateRepositoryGitData ( ctx context . Context , u * models . User , repo * models . Repository , opts migration . MigrateOptions ) ( * models . Repository , error ) {
2019-12-14 20:30:01 +03:00
repoPath := models . RepoPath ( u . Name , opts . RepoName )
if u . IsOrganization ( ) {
t , err := u . GetOwnerTeam ( )
if err != nil {
return nil , err
}
repo . NumWatches = t . NumMembers
} else {
repo . NumWatches = 1
}
migrateTimeout := time . Duration ( setting . Git . Timeout . Migrate ) * time . Second
var err error
2020-08-11 23:05:34 +03:00
if err = util . RemoveAll ( repoPath ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "Failed to remove %s: %v" , repoPath , err )
}
2020-12-02 21:36:06 +03:00
if err = git . CloneWithContext ( ctx , opts . CloneAddr , repoPath , git . CloneRepoOptions {
2019-12-14 20:30:01 +03:00
Mirror : true ,
Quiet : true ,
Timeout : migrateTimeout ,
} ) ; err != nil {
return repo , fmt . Errorf ( "Clone: %v" , err )
}
if opts . Wiki {
wikiPath := models . WikiPath ( u . Name , opts . RepoName )
2020-07-06 05:08:32 +03:00
wikiRemotePath := WikiRemoteURL ( opts . CloneAddr )
2019-12-14 20:30:01 +03:00
if len ( wikiRemotePath ) > 0 {
2020-08-11 23:05:34 +03:00
if err := util . RemoveAll ( wikiPath ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "Failed to remove %s: %v" , wikiPath , err )
}
2020-12-02 21:36:06 +03:00
if err = git . CloneWithContext ( ctx , wikiRemotePath , wikiPath , git . CloneRepoOptions {
2019-12-14 20:30:01 +03:00
Mirror : true ,
Quiet : true ,
Timeout : migrateTimeout ,
Branch : "master" ,
} ) ; err != nil {
log . Warn ( "Clone wiki: %v" , err )
2020-08-11 23:05:34 +03:00
if err := util . RemoveAll ( wikiPath ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "Failed to remove %s: %v" , wikiPath , err )
}
}
}
}
gitRepo , err := git . OpenRepository ( repoPath )
if err != nil {
return repo , fmt . Errorf ( "OpenRepository: %v" , err )
}
defer gitRepo . Close ( )
repo . IsEmpty , err = gitRepo . IsEmpty ( )
if err != nil {
return repo , fmt . Errorf ( "git.IsEmpty: %v" , err )
}
2020-09-15 17:37:44 +03:00
if ! repo . IsEmpty {
if len ( repo . DefaultBranch ) == 0 {
// Try to get HEAD branch and set it as default branch.
headBranch , err := gitRepo . GetHEADBranch ( )
if err != nil {
return repo , fmt . Errorf ( "GetHEADBranch: %v" , err )
}
if headBranch != nil {
repo . DefaultBranch = headBranch . Name
}
2019-12-14 20:30:01 +03:00
}
2020-09-15 17:37:44 +03:00
if ! opts . Releases {
if err = SyncReleasesWithTags ( repo , gitRepo ) ; err != nil {
log . Error ( "Failed to synchronize tags to releases for repository: %v" , err )
}
2019-12-14 20:30:01 +03:00
}
2021-04-09 01:25:57 +03:00
if opts . LFS {
ep := lfs . DetermineEndpoint ( opts . CloneAddr , opts . LFSEndpoint )
if err = StoreMissingLfsObjectsInRepository ( ctx , repo , gitRepo , ep ) ; err != nil {
log . Error ( "Failed to store missing LFS objects for repository: %v" , err )
}
}
2019-12-14 20:30:01 +03:00
}
2020-01-12 15:11:17 +03:00
if err = repo . UpdateSize ( models . DefaultDBContext ( ) ) ; err != nil {
2019-12-14 20:30:01 +03:00
log . Error ( "Failed to update size for repository: %v" , err )
}
if opts . Mirror {
2021-01-03 02:47:47 +03:00
mirrorModel := models . Mirror {
2019-12-14 20:30:01 +03:00
RepoID : repo . ID ,
Interval : setting . Mirror . DefaultInterval ,
EnablePrune : true ,
NextUpdateUnix : timeutil . TimeStampNow ( ) . AddDuration ( setting . Mirror . DefaultInterval ) ,
2021-04-09 01:25:57 +03:00
LFS : opts . LFS ,
}
if opts . LFS {
mirrorModel . LFSEndpoint = opts . LFSEndpoint
2021-01-03 02:47:47 +03:00
}
if opts . MirrorInterval != "" {
parsedInterval , err := time . ParseDuration ( opts . MirrorInterval )
if err != nil {
log . Error ( "Failed to set Interval: %v" , err )
return repo , err
}
if parsedInterval == 0 {
mirrorModel . Interval = 0
mirrorModel . NextUpdateUnix = 0
} else if parsedInterval < setting . Mirror . MinInterval {
err := fmt . Errorf ( "Interval %s is set below Minimum Interval of %s" , parsedInterval , setting . Mirror . MinInterval )
log . Error ( "Interval: %s is too frequent" , opts . MirrorInterval )
return repo , err
} else {
mirrorModel . Interval = parsedInterval
mirrorModel . NextUpdateUnix = timeutil . TimeStampNow ( ) . AddDuration ( parsedInterval )
}
}
if err = models . InsertMirror ( & mirrorModel ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "InsertOne: %v" , err )
}
repo . IsMirror = true
err = models . UpdateRepository ( repo , false )
} else {
repo , err = CleanUpMigrateInfo ( repo )
}
return repo , err
}
// cleanUpMigrateGitConfig removes mirror info which prevents "push --all".
// This also removes possible user credentials.
func cleanUpMigrateGitConfig ( configPath string ) error {
cfg , err := ini . Load ( configPath )
if err != nil {
return fmt . Errorf ( "open config file: %v" , err )
}
cfg . DeleteSection ( "remote \"origin\"" )
if err = cfg . SaveToIndent ( configPath , "\t" ) ; err != nil {
return fmt . Errorf ( "save config file: %v" , err )
}
return nil
}
// CleanUpMigrateInfo finishes migrating repository and/or wiki with things that don't need to be done for mirrors.
func CleanUpMigrateInfo ( repo * models . Repository ) ( * models . Repository , error ) {
repoPath := repo . RepoPath ( )
2020-01-20 23:01:19 +03:00
if err := createDelegateHooks ( repoPath ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "createDelegateHooks: %v" , err )
}
if repo . HasWiki ( ) {
2020-01-20 23:01:19 +03:00
if err := createDelegateHooks ( repo . WikiPath ( ) ) ; err != nil {
2019-12-14 20:30:01 +03:00
return repo , fmt . Errorf ( "createDelegateHooks.(wiki): %v" , err )
}
}
_ , err := git . NewCommand ( "remote" , "rm" , "origin" ) . RunInDir ( repoPath )
if err != nil && ! strings . HasPrefix ( err . Error ( ) , "exit status 128 - fatal: No such remote " ) {
return repo , fmt . Errorf ( "CleanUpMigrateInfo: %v" , err )
}
if repo . HasWiki ( ) {
if err := cleanUpMigrateGitConfig ( path . Join ( repo . WikiPath ( ) , "config" ) ) ; err != nil {
return repo , fmt . Errorf ( "cleanUpMigrateGitConfig (wiki): %v" , err )
}
}
return repo , models . UpdateRepository ( repo , false )
}
// SyncReleasesWithTags synchronizes release table with repository tags
func SyncReleasesWithTags ( repo * models . Repository , gitRepo * git . Repository ) error {
existingRelTags := make ( map [ string ] struct { } )
2020-01-24 22:00:29 +03:00
opts := models . FindReleasesOptions { IncludeDrafts : true , IncludeTags : true , ListOptions : models . ListOptions { PageSize : 50 } }
2019-12-14 20:30:01 +03:00
for page := 1 ; ; page ++ {
2020-01-24 22:00:29 +03:00
opts . Page = page
rels , err := models . GetReleasesByRepoID ( repo . ID , opts )
2019-12-14 20:30:01 +03:00
if err != nil {
return fmt . Errorf ( "GetReleasesByRepoID: %v" , err )
}
if len ( rels ) == 0 {
break
}
for _ , rel := range rels {
if rel . IsDraft {
continue
}
commitID , err := gitRepo . GetTagCommitID ( rel . TagName )
if err != nil && ! git . IsErrNotExist ( err ) {
2019-12-26 14:29:45 +03:00
return fmt . Errorf ( "GetTagCommitID: %s: %v" , rel . TagName , err )
2019-12-14 20:30:01 +03:00
}
if git . IsErrNotExist ( err ) || commitID != rel . Sha1 {
if err := models . PushUpdateDeleteTag ( repo , rel . TagName ) ; err != nil {
2019-12-26 14:29:45 +03:00
return fmt . Errorf ( "PushUpdateDeleteTag: %s: %v" , rel . TagName , err )
2019-12-14 20:30:01 +03:00
}
} else {
existingRelTags [ strings . ToLower ( rel . TagName ) ] = struct { } { }
}
}
}
tags , err := gitRepo . GetTags ( )
if err != nil {
return fmt . Errorf ( "GetTags: %v" , err )
}
for _ , tagName := range tags {
if _ , ok := existingRelTags [ strings . ToLower ( tagName ) ] ; ! ok {
2020-01-10 12:34:21 +03:00
if err := PushUpdateAddTag ( repo , gitRepo , tagName ) ; err != nil {
return fmt . Errorf ( "pushUpdateAddTag: %v" , err )
2019-12-14 20:30:01 +03:00
}
}
}
return nil
}
2020-01-10 12:34:21 +03:00
// PushUpdateAddTag must be called for any push actions to add tag
func PushUpdateAddTag ( repo * models . Repository , gitRepo * git . Repository , tagName string ) error {
tag , err := gitRepo . GetTag ( tagName )
if err != nil {
return fmt . Errorf ( "GetTag: %v" , err )
}
commit , err := tag . Commit ( )
if err != nil {
return fmt . Errorf ( "Commit: %v" , err )
}
sig := tag . Tagger
if sig == nil {
sig = commit . Author
}
if sig == nil {
sig = commit . Committer
}
var author * models . User
var createdAt = time . Unix ( 1 , 0 )
if sig != nil {
author , err = models . GetUserByEmail ( sig . Email )
if err != nil && ! models . IsErrUserNotExist ( err ) {
return fmt . Errorf ( "GetUserByEmail: %v" , err )
}
createdAt = sig . When
}
commitsCount , err := commit . CommitsCount ( )
if err != nil {
return fmt . Errorf ( "CommitsCount: %v" , err )
}
var rel = models . Release {
RepoID : repo . ID ,
TagName : tagName ,
LowerTagName : strings . ToLower ( tagName ) ,
Sha1 : commit . ID . String ( ) ,
NumCommits : commitsCount ,
CreatedUnix : timeutil . TimeStamp ( createdAt . Unix ( ) ) ,
IsTag : true ,
}
if author != nil {
rel . PublisherID = author . ID
}
return models . SaveOrUpdateTag ( repo , & rel )
}
2021-04-09 01:25:57 +03:00
// StoreMissingLfsObjectsInRepository downloads missing LFS objects
func StoreMissingLfsObjectsInRepository ( ctx context . Context , repo * models . Repository , gitRepo * git . Repository , endpoint * url . URL ) error {
client := lfs . NewClient ( endpoint )
contentStore := lfs . NewContentStore ( )
pointerChan := make ( chan lfs . PointerBlob )
errChan := make ( chan error , 1 )
go lfs . SearchPointerBlobs ( ctx , gitRepo , pointerChan , errChan )
err := func ( ) error {
for pointerBlob := range pointerChan {
meta , err := models . NewLFSMetaObject ( & models . LFSMetaObject { Pointer : pointerBlob . Pointer , RepositoryID : repo . ID } )
if err != nil {
return fmt . Errorf ( "StoreMissingLfsObjectsInRepository models.NewLFSMetaObject: %w" , err )
}
if meta . Existing {
continue
}
log . Trace ( "StoreMissingLfsObjectsInRepository: LFS OID[%s] not present in repository %s" , pointerBlob . Oid , repo . FullName ( ) )
err = func ( ) error {
exist , err := contentStore . Exists ( pointerBlob . Pointer )
if err != nil {
return fmt . Errorf ( "StoreMissingLfsObjectsInRepository contentStore.Exists: %w" , err )
}
if ! exist {
if setting . LFS . MaxFileSize > 0 && pointerBlob . Size > setting . LFS . MaxFileSize {
log . Info ( "LFS OID[%s] download denied because of LFS_MAX_FILE_SIZE=%d < size %d" , pointerBlob . Oid , setting . LFS . MaxFileSize , pointerBlob . Size )
return nil
}
stream , err := client . Download ( ctx , pointerBlob . Oid , pointerBlob . Size )
if err != nil {
return fmt . Errorf ( "StoreMissingLfsObjectsInRepository: LFS OID[%s] failed to download: %w" , pointerBlob . Oid , err )
}
defer stream . Close ( )
if err := contentStore . Put ( pointerBlob . Pointer , stream ) ; err != nil {
return fmt . Errorf ( "StoreMissingLfsObjectsInRepository LFS OID[%s] contentStore.Put: %w" , pointerBlob . Oid , err )
}
} else {
log . Trace ( "StoreMissingLfsObjectsInRepository: LFS OID[%s] already present in content store" , pointerBlob . Oid )
}
return nil
} ( )
if err != nil {
if _ , err2 := repo . RemoveLFSMetaObjectByOid ( meta . Oid ) ; err2 != nil {
log . Error ( "StoreMissingLfsObjectsInRepository RemoveLFSMetaObjectByOid[Oid: %s]: %w" , meta . Oid , err2 )
}
select {
case <- ctx . Done ( ) :
return nil
default :
}
return err
}
}
return nil
} ( )
if err != nil {
return err
}
err , has := <- errChan
if has {
return err
}
return nil
}