mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-29 10:57:44 +09:00 
			
		
		
		
	Move create/fork repository from models to modules/repository (#9489)
* Move create/fork repository from models to modules/repository * fix wrong reference * fix test * fix test * fix lint * Fix DBContext * remove duplicated TestMain * fix lint * fix conflicts
This commit is contained in:
		| @@ -22,9 +22,9 @@ var labelColorPattern = regexp.MustCompile("#([a-fA-F0-9]{6})") | |||||||
| // GetLabelTemplateFile loads the label template file by given name, | // GetLabelTemplateFile loads the label template file by given name, | ||||||
| // then parses and returns a list of name-color pairs and optionally description. | // then parses and returns a list of name-color pairs and optionally description. | ||||||
| func GetLabelTemplateFile(name string) ([][3]string, error) { | func GetLabelTemplateFile(name string) ([][3]string, error) { | ||||||
| 	data, err := getRepoInitFile("label", name) | 	data, err := GetRepoInitFile("label", name) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, fmt.Errorf("getRepoInitFile: %v", err) | 		return nil, fmt.Errorf("GetRepoInitFile: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	lines := strings.Split(string(data), "\n") | 	lines := strings.Split(string(data), "\n") | ||||||
| @@ -175,8 +175,8 @@ func initalizeLabels(e Engine, repoID int64, labelTemplate string) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // InitalizeLabels adds a label set to a repository using a template | // InitalizeLabels adds a label set to a repository using a template | ||||||
| func InitalizeLabels(repoID int64, labelTemplate string) error { | func InitalizeLabels(ctx DBContext, repoID int64, labelTemplate string) error { | ||||||
| 	return initalizeLabels(x, repoID, labelTemplate) | 	return initalizeLabels(ctx.e, repoID, labelTemplate) | ||||||
| } | } | ||||||
|  |  | ||||||
| func newLabel(e Engine, label *Label) error { | func newLabel(e Engine, label *Label) error { | ||||||
|   | |||||||
| @@ -5,12 +5,9 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/modules/structs" |  | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -377,133 +374,3 @@ func TestUsersInTeamsCount(t *testing.T) { | |||||||
| 	test([]int64{1, 2, 3, 4, 5}, []int64{2, 5}, 2)    // userid 2,4 | 	test([]int64{1, 2, 3, 4, 5}, []int64{2, 5}, 2)    // userid 2,4 | ||||||
| 	test([]int64{1, 2, 3, 4, 5}, []int64{2, 3, 5}, 3) // userid 2,4,5 | 	test([]int64{1, 2, 3, 4, 5}, []int64{2, 3, 5}, 3) // userid 2,4,5 | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestIncludesAllRepositoriesTeams(t *testing.T) { |  | ||||||
| 	assert.NoError(t, PrepareTestDatabase()) |  | ||||||
|  |  | ||||||
| 	testTeamRepositories := func(teamID int64, repoIds []int64) { |  | ||||||
| 		team := AssertExistsAndLoadBean(t, &Team{ID: teamID}).(*Team) |  | ||||||
| 		assert.NoError(t, team.GetRepositories(), "%s: GetRepositories", team.Name) |  | ||||||
| 		assert.Len(t, team.Repos, team.NumRepos, "%s: len repo", team.Name) |  | ||||||
| 		assert.Equal(t, len(repoIds), len(team.Repos), "%s: repo count", team.Name) |  | ||||||
| 		for i, rid := range repoIds { |  | ||||||
| 			if rid > 0 { |  | ||||||
| 				assert.True(t, team.HasRepository(rid), "%s: HasRepository(%d) %d", rid, i) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Get an admin user. |  | ||||||
| 	user, err := GetUserByID(1) |  | ||||||
| 	assert.NoError(t, err, "GetUserByID") |  | ||||||
|  |  | ||||||
| 	// Create org. |  | ||||||
| 	org := &User{ |  | ||||||
| 		Name:       "All repo", |  | ||||||
| 		IsActive:   true, |  | ||||||
| 		Type:       UserTypeOrganization, |  | ||||||
| 		Visibility: structs.VisibleTypePublic, |  | ||||||
| 	} |  | ||||||
| 	assert.NoError(t, CreateOrganization(org, user), "CreateOrganization") |  | ||||||
|  |  | ||||||
| 	// Check Owner team. |  | ||||||
| 	ownerTeam, err := org.GetOwnerTeam() |  | ||||||
| 	assert.NoError(t, err, "GetOwnerTeam") |  | ||||||
| 	assert.True(t, ownerTeam.IncludesAllRepositories, "Owner team includes all repositories") |  | ||||||
|  |  | ||||||
| 	// Create repos. |  | ||||||
| 	repoIds := make([]int64, 0) |  | ||||||
| 	for i := 0; i < 3; i++ { |  | ||||||
| 		r, err := CreateRepository(user, org, CreateRepoOptions{Name: fmt.Sprintf("repo-%d", i)}) |  | ||||||
| 		assert.NoError(t, err, "CreateRepository %d", i) |  | ||||||
| 		if r != nil { |  | ||||||
| 			repoIds = append(repoIds, r.ID) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	// Get fresh copy of Owner team after creating repos. |  | ||||||
| 	ownerTeam, err = org.GetOwnerTeam() |  | ||||||
| 	assert.NoError(t, err, "GetOwnerTeam") |  | ||||||
|  |  | ||||||
| 	// Create teams and check repositories. |  | ||||||
| 	teams := []*Team{ |  | ||||||
| 		ownerTeam, |  | ||||||
| 		{ |  | ||||||
| 			OrgID:                   org.ID, |  | ||||||
| 			Name:                    "team one", |  | ||||||
| 			Authorize:               AccessModeRead, |  | ||||||
| 			IncludesAllRepositories: true, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			OrgID:                   org.ID, |  | ||||||
| 			Name:                    "team 2", |  | ||||||
| 			Authorize:               AccessModeRead, |  | ||||||
| 			IncludesAllRepositories: false, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			OrgID:                   org.ID, |  | ||||||
| 			Name:                    "team three", |  | ||||||
| 			Authorize:               AccessModeWrite, |  | ||||||
| 			IncludesAllRepositories: true, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			OrgID:                   org.ID, |  | ||||||
| 			Name:                    "team 4", |  | ||||||
| 			Authorize:               AccessModeWrite, |  | ||||||
| 			IncludesAllRepositories: false, |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
| 	teamRepos := [][]int64{ |  | ||||||
| 		repoIds, |  | ||||||
| 		repoIds, |  | ||||||
| 		{}, |  | ||||||
| 		repoIds, |  | ||||||
| 		{}, |  | ||||||
| 	} |  | ||||||
| 	for i, team := range teams { |  | ||||||
| 		if i > 0 { // first team is Owner. |  | ||||||
| 			assert.NoError(t, NewTeam(team), "%s: NewTeam", team.Name) |  | ||||||
| 		} |  | ||||||
| 		testTeamRepositories(team.ID, teamRepos[i]) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Update teams and check repositories. |  | ||||||
| 	teams[3].IncludesAllRepositories = false |  | ||||||
| 	teams[4].IncludesAllRepositories = true |  | ||||||
| 	teamRepos[4] = repoIds |  | ||||||
| 	for i, team := range teams { |  | ||||||
| 		assert.NoError(t, UpdateTeam(team, false, true), "%s: UpdateTeam", team.Name) |  | ||||||
| 		testTeamRepositories(team.ID, teamRepos[i]) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Create repo and check teams repositories. |  | ||||||
| 	org.Teams = nil // Reset teams to allow their reloading. |  | ||||||
| 	r, err := CreateRepository(user, org, CreateRepoOptions{Name: "repo-last"}) |  | ||||||
| 	assert.NoError(t, err, "CreateRepository last") |  | ||||||
| 	if r != nil { |  | ||||||
| 		repoIds = append(repoIds, r.ID) |  | ||||||
| 	} |  | ||||||
| 	teamRepos[0] = repoIds |  | ||||||
| 	teamRepos[1] = repoIds |  | ||||||
| 	teamRepos[4] = repoIds |  | ||||||
| 	for i, team := range teams { |  | ||||||
| 		testTeamRepositories(team.ID, teamRepos[i]) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove repo and check teams repositories. |  | ||||||
| 	assert.NoError(t, DeleteRepository(user, org.ID, repoIds[0]), "DeleteRepository") |  | ||||||
| 	teamRepos[0] = repoIds[1:] |  | ||||||
| 	teamRepos[1] = repoIds[1:] |  | ||||||
| 	teamRepos[3] = repoIds[1:3] |  | ||||||
| 	teamRepos[4] = repoIds[1:] |  | ||||||
| 	for i, team := range teams { |  | ||||||
| 		testTeamRepositories(team.ID, teamRepos[i]) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Wipe created items. |  | ||||||
| 	for i, rid := range repoIds { |  | ||||||
| 		if i > 0 { // first repo already deleted. |  | ||||||
| 			assert.NoError(t, DeleteRepository(user, org.ID, rid), "DeleteRepository %d", i) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	assert.NoError(t, DeleteOrganization(org), "DeleteOrganization") |  | ||||||
| } |  | ||||||
|   | |||||||
							
								
								
									
										400
									
								
								models/repo.go
									
									
									
									
									
								
							
							
						
						
									
										400
									
								
								models/repo.go
									
									
									
									
									
								
							| @@ -6,7 +6,6 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bytes" |  | ||||||
| 	"context" | 	"context" | ||||||
| 	"crypto/md5" | 	"crypto/md5" | ||||||
| 	"errors" | 	"errors" | ||||||
| @@ -38,7 +37,6 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/timeutil" | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
| 	"code.gitea.io/gitea/modules/util" | 	"code.gitea.io/gitea/modules/util" | ||||||
|  |  | ||||||
| 	"github.com/mcuadros/go-version" |  | ||||||
| 	"github.com/unknwon/com" | 	"github.com/unknwon/com" | ||||||
| 	"xorm.io/builder" | 	"xorm.io/builder" | ||||||
| ) | ) | ||||||
| @@ -715,7 +713,7 @@ func (repo *Repository) IsOwnedBy(userID int64) bool { | |||||||
| func (repo *Repository) updateSize(e Engine) error { | func (repo *Repository) updateSize(e Engine) error { | ||||||
| 	size, err := util.GetDirectorySize(repo.RepoPath()) | 	size, err := util.GetDirectorySize(repo.RepoPath()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("UpdateSize: %v", err) | 		return fmt.Errorf("updateSize: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repo.Size = size | 	repo.Size = size | ||||||
| @@ -724,8 +722,8 @@ func (repo *Repository) updateSize(e Engine) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateSize updates the repository size, calculating it using util.GetDirectorySize | // UpdateSize updates the repository size, calculating it using util.GetDirectorySize | ||||||
| func (repo *Repository) UpdateSize() error { | func (repo *Repository) UpdateSize(ctx DBContext) error { | ||||||
| 	return repo.updateSize(x) | 	return repo.updateSize(ctx.e) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CanUserFork returns true if specified user can fork repository. | // CanUserFork returns true if specified user can fork repository. | ||||||
| @@ -966,64 +964,6 @@ func createDelegateHooks(repoPath string) (err error) { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // initRepoCommit temporarily changes with work directory. |  | ||||||
| func initRepoCommit(tmpPath string, repo *Repository, u *User) (err error) { |  | ||||||
| 	commitTimeStr := time.Now().Format(time.RFC3339) |  | ||||||
|  |  | ||||||
| 	sig := u.NewGitSig() |  | ||||||
| 	// Because this may call hooks we should pass in the environment |  | ||||||
| 	env := append(os.Environ(), |  | ||||||
| 		"GIT_AUTHOR_NAME="+sig.Name, |  | ||||||
| 		"GIT_AUTHOR_EMAIL="+sig.Email, |  | ||||||
| 		"GIT_AUTHOR_DATE="+commitTimeStr, |  | ||||||
| 		"GIT_COMMITTER_NAME="+sig.Name, |  | ||||||
| 		"GIT_COMMITTER_EMAIL="+sig.Email, |  | ||||||
| 		"GIT_COMMITTER_DATE="+commitTimeStr, |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	if stdout, err := git.NewCommand("add", "--all"). |  | ||||||
| 		SetDescription(fmt.Sprintf("initRepoCommit (git add): %s", tmpPath)). |  | ||||||
| 		RunInDir(tmpPath); err != nil { |  | ||||||
| 		log.Error("git add --all failed: Stdout: %s\nError: %v", stdout, err) |  | ||||||
| 		return fmt.Errorf("git add --all: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	binVersion, err := git.BinVersion() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("Unable to get git version: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	args := []string{ |  | ||||||
| 		"commit", fmt.Sprintf("--author='%s <%s>'", sig.Name, sig.Email), |  | ||||||
| 		"-m", "Initial commit", |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if version.Compare(binVersion, "1.7.9", ">=") { |  | ||||||
| 		sign, keyID := SignInitialCommit(tmpPath, u) |  | ||||||
| 		if sign { |  | ||||||
| 			args = append(args, "-S"+keyID) |  | ||||||
| 		} else if version.Compare(binVersion, "2.0.0", ">=") { |  | ||||||
| 			args = append(args, "--no-gpg-sign") |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if stdout, err := git.NewCommand(args...). |  | ||||||
| 		SetDescription(fmt.Sprintf("initRepoCommit (git commit): %s", tmpPath)). |  | ||||||
| 		RunInDirWithEnv(tmpPath, env); err != nil { |  | ||||||
| 		log.Error("Failed to commit: %v: Stdout: %s\nError: %v", args, stdout, err) |  | ||||||
| 		return fmt.Errorf("git commit: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if stdout, err := git.NewCommand("push", "origin", "master"). |  | ||||||
| 		SetDescription(fmt.Sprintf("initRepoCommit (git push): %s", tmpPath)). |  | ||||||
| 		RunInDirWithEnv(tmpPath, InternalPushingEnvironment(u, repo)); err != nil { |  | ||||||
| 		log.Error("Failed to push back to master: Stdout: %s\nError: %v", stdout, err) |  | ||||||
| 		return fmt.Errorf("git push: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // CreateRepoOptions contains the create repository options | // CreateRepoOptions contains the create repository options | ||||||
| type CreateRepoOptions struct { | type CreateRepoOptions struct { | ||||||
| 	Name           string | 	Name           string | ||||||
| @@ -1040,7 +980,8 @@ type CreateRepoOptions struct { | |||||||
| 	Status         RepositoryStatus | 	Status         RepositoryStatus | ||||||
| } | } | ||||||
|  |  | ||||||
| func getRepoInitFile(tp, name string) ([]byte, error) { | // GetRepoInitFile returns repository init files | ||||||
|  | func GetRepoInitFile(tp, name string) ([]byte, error) { | ||||||
| 	cleanedName := strings.TrimLeft(path.Clean("/"+name), "/") | 	cleanedName := strings.TrimLeft(path.Clean("/"+name), "/") | ||||||
| 	relPath := path.Join("options", tp, cleanedName) | 	relPath := path.Join("options", tp, cleanedName) | ||||||
|  |  | ||||||
| @@ -1064,140 +1005,6 @@ func getRepoInitFile(tp, name string) ([]byte, error) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func prepareRepoCommit(e Engine, repo *Repository, tmpDir, repoPath string, opts CreateRepoOptions) error { |  | ||||||
| 	commitTimeStr := time.Now().Format(time.RFC3339) |  | ||||||
| 	authorSig := repo.Owner.NewGitSig() |  | ||||||
|  |  | ||||||
| 	// Because this may call hooks we should pass in the environment |  | ||||||
| 	env := append(os.Environ(), |  | ||||||
| 		"GIT_AUTHOR_NAME="+authorSig.Name, |  | ||||||
| 		"GIT_AUTHOR_EMAIL="+authorSig.Email, |  | ||||||
| 		"GIT_AUTHOR_DATE="+commitTimeStr, |  | ||||||
| 		"GIT_COMMITTER_NAME="+authorSig.Name, |  | ||||||
| 		"GIT_COMMITTER_EMAIL="+authorSig.Email, |  | ||||||
| 		"GIT_COMMITTER_DATE="+commitTimeStr, |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	// Clone to temporary path and do the init commit. |  | ||||||
| 	if stdout, err := git.NewCommand("clone", repoPath, tmpDir). |  | ||||||
| 		SetDescription(fmt.Sprintf("initRepository (git clone): %s to %s", repoPath, tmpDir)). |  | ||||||
| 		RunInDirWithEnv("", env); err != nil { |  | ||||||
| 		log.Error("Failed to clone from %v into %s: stdout: %s\nError: %v", repo, tmpDir, stdout, err) |  | ||||||
| 		return fmt.Errorf("git clone: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// README |  | ||||||
| 	data, err := getRepoInitFile("readme", opts.Readme) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("getRepoInitFile[%s]: %v", opts.Readme, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	cloneLink := repo.cloneLink(false) |  | ||||||
| 	match := map[string]string{ |  | ||||||
| 		"Name":           repo.Name, |  | ||||||
| 		"Description":    repo.Description, |  | ||||||
| 		"CloneURL.SSH":   cloneLink.SSH, |  | ||||||
| 		"CloneURL.HTTPS": cloneLink.HTTPS, |  | ||||||
| 	} |  | ||||||
| 	if err = ioutil.WriteFile(filepath.Join(tmpDir, "README.md"), |  | ||||||
| 		[]byte(com.Expand(string(data), match)), 0644); err != nil { |  | ||||||
| 		return fmt.Errorf("write README.md: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// .gitignore |  | ||||||
| 	if len(opts.Gitignores) > 0 { |  | ||||||
| 		var buf bytes.Buffer |  | ||||||
| 		names := strings.Split(opts.Gitignores, ",") |  | ||||||
| 		for _, name := range names { |  | ||||||
| 			data, err = getRepoInitFile("gitignore", name) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return fmt.Errorf("getRepoInitFile[%s]: %v", name, err) |  | ||||||
| 			} |  | ||||||
| 			buf.WriteString("# ---> " + name + "\n") |  | ||||||
| 			buf.Write(data) |  | ||||||
| 			buf.WriteString("\n") |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if buf.Len() > 0 { |  | ||||||
| 			if err = ioutil.WriteFile(filepath.Join(tmpDir, ".gitignore"), buf.Bytes(), 0644); err != nil { |  | ||||||
| 				return fmt.Errorf("write .gitignore: %v", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// LICENSE |  | ||||||
| 	if len(opts.License) > 0 { |  | ||||||
| 		data, err = getRepoInitFile("license", opts.License) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("getRepoInitFile[%s]: %v", opts.License, err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err = ioutil.WriteFile(filepath.Join(tmpDir, "LICENSE"), data, 0644); err != nil { |  | ||||||
| 			return fmt.Errorf("write LICENSE: %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func checkInitRepository(repoPath string) (err error) { |  | ||||||
| 	// Somehow the directory could exist. |  | ||||||
| 	if com.IsExist(repoPath) { |  | ||||||
| 		return fmt.Errorf("initRepository: path already exists: %s", repoPath) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Init git bare new repository. |  | ||||||
| 	if err = git.InitRepository(repoPath, true); err != nil { |  | ||||||
| 		return fmt.Errorf("InitRepository: %v", err) |  | ||||||
| 	} else if err = createDelegateHooks(repoPath); err != nil { |  | ||||||
| 		return fmt.Errorf("createDelegateHooks: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // InitRepository initializes README and .gitignore if needed. |  | ||||||
| func initRepository(e Engine, repoPath string, u *User, repo *Repository, opts CreateRepoOptions) (err error) { |  | ||||||
| 	if err = checkInitRepository(repoPath); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Initialize repository according to user's choice. |  | ||||||
| 	if opts.AutoInit { |  | ||||||
| 		tmpDir, err := ioutil.TempDir(os.TempDir(), "gitea-"+repo.Name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("Failed to create temp dir for repository %s: %v", repo.RepoPath(), err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		defer os.RemoveAll(tmpDir) |  | ||||||
|  |  | ||||||
| 		if err = prepareRepoCommit(e, repo, tmpDir, repoPath, opts); err != nil { |  | ||||||
| 			return fmt.Errorf("prepareRepoCommit: %v", err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Apply changes and commit. |  | ||||||
| 		if err = initRepoCommit(tmpDir, repo, u); err != nil { |  | ||||||
| 			return fmt.Errorf("initRepoCommit: %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Re-fetch the repository from database before updating it (else it would |  | ||||||
| 	// override changes that were done earlier with sql) |  | ||||||
| 	if repo, err = getRepositoryByID(e, repo.ID); err != nil { |  | ||||||
| 		return fmt.Errorf("getRepositoryByID: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if !opts.AutoInit { |  | ||||||
| 		repo.IsEmpty = true |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo.DefaultBranch = "master" |  | ||||||
| 	if err = updateRepository(e, repo, false); err != nil { |  | ||||||
| 		return fmt.Errorf("updateRepository: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	reservedRepoNames    = []string{".", ".."} | 	reservedRepoNames    = []string{".", ".."} | ||||||
| 	reservedRepoPatterns = []string{"*.git", "*.wiki"} | 	reservedRepoPatterns = []string{"*.git", "*.wiki"} | ||||||
| @@ -1208,22 +1015,23 @@ func IsUsableRepoName(name string) error { | |||||||
| 	return isUsableName(reservedRepoNames, reservedRepoPatterns, name) | 	return isUsableName(reservedRepoNames, reservedRepoPatterns, name) | ||||||
| } | } | ||||||
|  |  | ||||||
| func createRepository(e Engine, doer, u *User, repo *Repository) (err error) { | // CreateRepository creates a repository for the user/organization. | ||||||
|  | func CreateRepository(ctx DBContext, doer, u *User, repo *Repository) (err error) { | ||||||
| 	if err = IsUsableRepoName(repo.Name); err != nil { | 	if err = IsUsableRepoName(repo.Name); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	has, err := isRepositoryExist(e, u, repo.Name) | 	has, err := isRepositoryExist(ctx.e, u, repo.Name) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("IsRepositoryExist: %v", err) | 		return fmt.Errorf("IsRepositoryExist: %v", err) | ||||||
| 	} else if has { | 	} else if has { | ||||||
| 		return ErrRepoAlreadyExist{u.Name, repo.Name} | 		return ErrRepoAlreadyExist{u.Name, repo.Name} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if _, err = e.Insert(repo); err != nil { | 	if _, err = ctx.e.Insert(repo); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	if err = deleteRepoRedirect(e, u.ID, repo.Name); err != nil { | 	if err = deleteRepoRedirect(ctx.e, u.ID, repo.Name); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -1252,20 +1060,19 @@ func createRepository(e Engine, doer, u *User, repo *Repository) (err error) { | |||||||
| 				Type:   tp, | 				Type:   tp, | ||||||
| 			}) | 			}) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if _, err = e.Insert(&units); err != nil { | 	if _, err = ctx.e.Insert(&units); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Remember visibility preference. | 	// Remember visibility preference. | ||||||
| 	u.LastRepoVisibility = repo.IsPrivate | 	u.LastRepoVisibility = repo.IsPrivate | ||||||
| 	if err = updateUserCols(e, u, "last_repo_visibility"); err != nil { | 	if err = updateUserCols(ctx.e, u, "last_repo_visibility"); err != nil { | ||||||
| 		return fmt.Errorf("updateUser: %v", err) | 		return fmt.Errorf("updateUser: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if _, err = e.Incr("num_repos").ID(u.ID).Update(new(User)); err != nil { | 	if _, err = ctx.e.Incr("num_repos").ID(u.ID).Update(new(User)); err != nil { | ||||||
| 		return fmt.Errorf("increment user total_repos: %v", err) | 		return fmt.Errorf("increment user total_repos: %v", err) | ||||||
| 	} | 	} | ||||||
| 	u.NumRepos++ | 	u.NumRepos++ | ||||||
| @@ -1277,107 +1084,41 @@ func createRepository(e Engine, doer, u *User, repo *Repository) (err error) { | |||||||
| 		} | 		} | ||||||
| 		for _, t := range u.Teams { | 		for _, t := range u.Teams { | ||||||
| 			if t.IncludesAllRepositories { | 			if t.IncludesAllRepositories { | ||||||
| 				if err := t.addRepository(e, repo); err != nil { | 				if err := t.addRepository(ctx.e, repo); err != nil { | ||||||
| 					return fmt.Errorf("addRepository: %v", err) | 					return fmt.Errorf("addRepository: %v", err) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if isAdmin, err := isUserRepoAdmin(e, repo, doer); err != nil { | 		if isAdmin, err := isUserRepoAdmin(ctx.e, repo, doer); err != nil { | ||||||
| 			return fmt.Errorf("isUserRepoAdmin: %v", err) | 			return fmt.Errorf("isUserRepoAdmin: %v", err) | ||||||
| 		} else if !isAdmin { | 		} else if !isAdmin { | ||||||
| 			// Make creator repo admin if it wan't assigned automatically | 			// Make creator repo admin if it wan't assigned automatically | ||||||
| 			if err = repo.addCollaborator(e, doer); err != nil { | 			if err = repo.addCollaborator(ctx.e, doer); err != nil { | ||||||
| 				return fmt.Errorf("AddCollaborator: %v", err) | 				return fmt.Errorf("AddCollaborator: %v", err) | ||||||
| 			} | 			} | ||||||
| 			if err = repo.changeCollaborationAccessMode(e, doer.ID, AccessModeAdmin); err != nil { | 			if err = repo.changeCollaborationAccessMode(ctx.e, doer.ID, AccessModeAdmin); err != nil { | ||||||
| 				return fmt.Errorf("ChangeCollaborationAccessMode: %v", err) | 				return fmt.Errorf("ChangeCollaborationAccessMode: %v", err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} else if err = repo.recalculateAccesses(e); err != nil { | 	} else if err = repo.recalculateAccesses(ctx.e); err != nil { | ||||||
| 		// Organization automatically called this in addRepository method. | 		// Organization automatically called this in addRepository method. | ||||||
| 		return fmt.Errorf("recalculateAccesses: %v", err) | 		return fmt.Errorf("recalculateAccesses: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if setting.Service.AutoWatchNewRepos { | 	if setting.Service.AutoWatchNewRepos { | ||||||
| 		if err = watchRepo(e, doer.ID, repo.ID, true); err != nil { | 		if err = watchRepo(ctx.e, doer.ID, repo.ID, true); err != nil { | ||||||
| 			return fmt.Errorf("watchRepo: %v", err) | 			return fmt.Errorf("watchRepo: %v", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err = copyDefaultWebhooksToRepo(e, repo.ID); err != nil { | 	if err = copyDefaultWebhooksToRepo(ctx.e, repo.ID); err != nil { | ||||||
| 		return fmt.Errorf("copyDefaultWebhooksToRepo: %v", err) | 		return fmt.Errorf("copyDefaultWebhooksToRepo: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateRepository creates a repository for the user/organization. |  | ||||||
| func CreateRepository(doer, u *User, opts CreateRepoOptions) (_ *Repository, err error) { |  | ||||||
| 	if !doer.IsAdmin && !u.CanCreateRepo() { |  | ||||||
| 		return nil, ErrReachLimitOfRepo{u.MaxRepoCreation} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo := &Repository{ |  | ||||||
| 		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, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	sess := x.NewSession() |  | ||||||
| 	defer sess.Close() |  | ||||||
| 	if err = sess.Begin(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = createRepository(sess, doer, u, repo); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// No need for init mirror. |  | ||||||
| 	if !opts.IsMirror { |  | ||||||
| 		repoPath := RepoPath(u.Name, repo.Name) |  | ||||||
| 		if err = initRepository(sess, repoPath, u, repo, opts); err != nil { |  | ||||||
| 			if err2 := os.RemoveAll(repoPath); err2 != nil { |  | ||||||
| 				log.Error("initRepository: %v", err) |  | ||||||
| 				return nil, fmt.Errorf( |  | ||||||
| 					"delete repo directory %s/%s failed(2): %v", u.Name, repo.Name, err2) |  | ||||||
| 			} |  | ||||||
| 			return nil, fmt.Errorf("initRepository: %v", err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Initialize Issue Labels if selected |  | ||||||
| 		if len(opts.IssueLabels) > 0 { |  | ||||||
| 			if err = initalizeLabels(sess, repo.ID, opts.IssueLabels); err != nil { |  | ||||||
| 				return nil, fmt.Errorf("initalizeLabels: %v", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if stdout, err := git.NewCommand("update-server-info"). |  | ||||||
| 			SetDescription(fmt.Sprintf("CreateRepository(git update-server-info): %s", repoPath)). |  | ||||||
| 			RunInDir(repoPath); err != nil { |  | ||||||
| 			log.Error("CreateRepitory(git update-server-info) in %v: Stdout: %s\nError: %v", repo, stdout, err) |  | ||||||
| 			return nil, fmt.Errorf("CreateRepository(git update-server-info): %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = sess.Commit(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return repo, err |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func countRepositories(userID int64, private bool) int64 { | func countRepositories(userID int64, private bool) int64 { | ||||||
| 	sess := x.Where("id > 0") | 	sess := x.Where("id > 0") | ||||||
|  |  | ||||||
| @@ -1414,6 +1155,12 @@ func RepoPath(userName, repoName string) string { | |||||||
| 	return filepath.Join(UserPath(userName), strings.ToLower(repoName)+".git") | 	return filepath.Join(UserPath(userName), strings.ToLower(repoName)+".git") | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // IncrementRepoForkNum increment repository fork number | ||||||
|  | func IncrementRepoForkNum(ctx DBContext, repoID int64) error { | ||||||
|  | 	_, err := ctx.e.Exec("UPDATE `repository` SET num_forks=num_forks+1 WHERE id=?", repoID) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  |  | ||||||
| // TransferOwnership transfers all corresponding setting from old user to new one. | // TransferOwnership transfers all corresponding setting from old user to new one. | ||||||
| func TransferOwnership(doer *User, newOwnerName string, repo *Repository) error { | func TransferOwnership(doer *User, newOwnerName string, repo *Repository) error { | ||||||
| 	newOwner, err := GetUserByName(newOwnerName) | 	newOwner, err := GetUserByName(newOwnerName) | ||||||
| @@ -1672,6 +1419,11 @@ func updateRepository(e Engine, repo *Repository, visibilityChanged bool) (err e | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // UpdateRepositoryCtx updates a repository with db context | ||||||
|  | func UpdateRepositoryCtx(ctx DBContext, repo *Repository, visibilityChanged bool) error { | ||||||
|  | 	return updateRepository(ctx.e, repo, visibilityChanged) | ||||||
|  | } | ||||||
|  |  | ||||||
| // UpdateRepository updates a repository | // UpdateRepository updates a repository | ||||||
| func UpdateRepository(repo *Repository, visibilityChanged bool) (err error) { | func UpdateRepository(repo *Repository, visibilityChanged bool) (err error) { | ||||||
| 	sess := x.NewSession() | 	sess := x.NewSession() | ||||||
| @@ -1987,6 +1739,11 @@ func GetRepositoryByID(id int64) (*Repository, error) { | |||||||
| 	return getRepositoryByID(x, id) | 	return getRepositoryByID(x, id) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetRepositoryByIDCtx returns the repository by given id if exists. | ||||||
|  | func GetRepositoryByIDCtx(ctx DBContext, id int64) (*Repository, error) { | ||||||
|  | 	return getRepositoryByID(ctx.e, id) | ||||||
|  | } | ||||||
|  |  | ||||||
| // GetRepositoriesMapByIDs returns the repositories by given id slice. | // GetRepositoriesMapByIDs returns the repositories by given id slice. | ||||||
| func GetRepositoriesMapByIDs(ids []int64) (map[int64]*Repository, error) { | func GetRepositoriesMapByIDs(ids []int64) (map[int64]*Repository, error) { | ||||||
| 	var repos = make(map[int64]*Repository, len(ids)) | 	var repos = make(map[int64]*Repository, len(ids)) | ||||||
| @@ -2436,20 +2193,16 @@ func HasForkedRepo(ownerID, repoID int64) (*Repository, bool) { | |||||||
| } | } | ||||||
|  |  | ||||||
| // CopyLFS copies LFS data from one repo to another | // CopyLFS copies LFS data from one repo to another | ||||||
| func CopyLFS(newRepo, oldRepo *Repository) error { | func CopyLFS(ctx DBContext, newRepo, oldRepo *Repository) error { | ||||||
| 	return copyLFS(x, newRepo, oldRepo) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func copyLFS(e Engine, newRepo, oldRepo *Repository) error { |  | ||||||
| 	var lfsObjects []*LFSMetaObject | 	var lfsObjects []*LFSMetaObject | ||||||
| 	if err := e.Where("repository_id=?", oldRepo.ID).Find(&lfsObjects); err != nil { | 	if err := ctx.e.Where("repository_id=?", oldRepo.ID).Find(&lfsObjects); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, v := range lfsObjects { | 	for _, v := range lfsObjects { | ||||||
| 		v.ID = 0 | 		v.ID = 0 | ||||||
| 		v.RepositoryID = newRepo.ID | 		v.RepositoryID = newRepo.ID | ||||||
| 		if _, err := e.Insert(v); err != nil { | 		if _, err := ctx.e.Insert(v); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -2457,81 +2210,6 @@ func copyLFS(e Engine, newRepo, oldRepo *Repository) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // ForkRepository forks a repository |  | ||||||
| func ForkRepository(doer, owner *User, oldRepo *Repository, name, desc string) (_ *Repository, err error) { |  | ||||||
| 	forkedRepo, err := oldRepo.GetUserFork(owner.ID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	if forkedRepo != nil { |  | ||||||
| 		return nil, ErrForkAlreadyExist{ |  | ||||||
| 			Uname:    owner.Name, |  | ||||||
| 			RepoName: oldRepo.FullName(), |  | ||||||
| 			ForkName: forkedRepo.FullName(), |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo := &Repository{ |  | ||||||
| 		OwnerID:       owner.ID, |  | ||||||
| 		Owner:         owner, |  | ||||||
| 		OwnerName:     owner.Name, |  | ||||||
| 		Name:          name, |  | ||||||
| 		LowerName:     strings.ToLower(name), |  | ||||||
| 		Description:   desc, |  | ||||||
| 		DefaultBranch: oldRepo.DefaultBranch, |  | ||||||
| 		IsPrivate:     oldRepo.IsPrivate, |  | ||||||
| 		IsEmpty:       oldRepo.IsEmpty, |  | ||||||
| 		IsFork:        true, |  | ||||||
| 		ForkID:        oldRepo.ID, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	sess := x.NewSession() |  | ||||||
| 	defer sess.Close() |  | ||||||
| 	if err = sess.Begin(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = createRepository(sess, doer, owner, repo); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err = sess.Exec("UPDATE `repository` SET num_forks=num_forks+1 WHERE id=?", oldRepo.ID); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repoPath := RepoPath(owner.Name, repo.Name) |  | ||||||
| 	if stdout, err := git.NewCommand( |  | ||||||
| 		"clone", "--bare", oldRepo.RepoPath(), repoPath). |  | ||||||
| 		SetDescription(fmt.Sprintf("ForkRepository(git clone): %s to %s", oldRepo.FullName(), repo.FullName())). |  | ||||||
| 		RunInDirTimeout(10*time.Minute, ""); err != nil { |  | ||||||
| 		log.Error("Fork Repository (git clone) Failed for %v (from %v):\nStdout: %s\nError: %v", repo, oldRepo, stdout, err) |  | ||||||
| 		return nil, fmt.Errorf("git clone: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if stdout, err := git.NewCommand("update-server-info"). |  | ||||||
| 		SetDescription(fmt.Sprintf("ForkRepository(git update-server-info): %s", repo.FullName())). |  | ||||||
| 		RunInDir(repoPath); err != nil { |  | ||||||
| 		log.Error("Fork Repository (git update-server-info) failed for %v:\nStdout: %s\nError: %v", repo, stdout, err) |  | ||||||
| 		return nil, fmt.Errorf("git update-server-info: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = createDelegateHooks(repoPath); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("createDelegateHooks: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	//Commit repo to get Fork ID |  | ||||||
| 	err = sess.Commit() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = repo.UpdateSize(); err != nil { |  | ||||||
| 		log.Error("Failed to update size for repository: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return repo, CopyLFS(repo, oldRepo) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetForks returns all the forks of the repository | // GetForks returns all the forks of the repository | ||||||
| func (repo *Repository) GetForks() ([]*Repository, error) { | func (repo *Repository) GetForks() ([]*Repository, error) { | ||||||
| 	forks := make([]*Repository, 0, repo.NumForks) | 	forks := make([]*Repository, 0, repo.NumForks) | ||||||
|   | |||||||
| @@ -5,14 +5,8 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
| 	"io/ioutil" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| @@ -71,186 +65,6 @@ func (gt GiteaTemplate) Globs() []glob.Glob { | |||||||
| 	return gt.globs | 	return gt.globs | ||||||
| } | } | ||||||
|  |  | ||||||
| func checkGiteaTemplate(tmpDir string) (*GiteaTemplate, error) { |  | ||||||
| 	gtPath := filepath.Join(tmpDir, ".gitea", "template") |  | ||||||
| 	if _, err := os.Stat(gtPath); os.IsNotExist(err) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} else if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	content, err := ioutil.ReadFile(gtPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	gt := &GiteaTemplate{ |  | ||||||
| 		Path:    gtPath, |  | ||||||
| 		Content: content, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return gt, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func generateRepoCommit(e Engine, repo, templateRepo, generateRepo *Repository, tmpDir string) error { |  | ||||||
| 	commitTimeStr := time.Now().Format(time.RFC3339) |  | ||||||
| 	authorSig := repo.Owner.NewGitSig() |  | ||||||
|  |  | ||||||
| 	// Because this may call hooks we should pass in the environment |  | ||||||
| 	env := append(os.Environ(), |  | ||||||
| 		"GIT_AUTHOR_NAME="+authorSig.Name, |  | ||||||
| 		"GIT_AUTHOR_EMAIL="+authorSig.Email, |  | ||||||
| 		"GIT_AUTHOR_DATE="+commitTimeStr, |  | ||||||
| 		"GIT_COMMITTER_NAME="+authorSig.Name, |  | ||||||
| 		"GIT_COMMITTER_EMAIL="+authorSig.Email, |  | ||||||
| 		"GIT_COMMITTER_DATE="+commitTimeStr, |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	// Clone to temporary path and do the init commit. |  | ||||||
| 	templateRepoPath := templateRepo.RepoPath() |  | ||||||
| 	if err := git.Clone(templateRepoPath, tmpDir, git.CloneRepoOptions{ |  | ||||||
| 		Depth: 1, |  | ||||||
| 	}); err != nil { |  | ||||||
| 		return fmt.Errorf("git clone: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := os.RemoveAll(path.Join(tmpDir, ".git")); err != nil { |  | ||||||
| 		return fmt.Errorf("remove git dir: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Variable expansion |  | ||||||
| 	gt, err := checkGiteaTemplate(tmpDir) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("checkGiteaTemplate: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if gt != nil { |  | ||||||
| 		if err := os.Remove(gt.Path); err != nil { |  | ||||||
| 			return fmt.Errorf("remove .giteatemplate: %v", err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Avoid walking tree if there are no globs |  | ||||||
| 		if len(gt.Globs()) > 0 { |  | ||||||
| 			tmpDirSlash := strings.TrimSuffix(filepath.ToSlash(tmpDir), "/") + "/" |  | ||||||
| 			if err := filepath.Walk(tmpDirSlash, func(path string, info os.FileInfo, walkErr error) error { |  | ||||||
| 				if walkErr != nil { |  | ||||||
| 					return walkErr |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				if info.IsDir() { |  | ||||||
| 					return nil |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				base := strings.TrimPrefix(filepath.ToSlash(path), tmpDirSlash) |  | ||||||
| 				for _, g := range gt.Globs() { |  | ||||||
| 					if g.Match(base) { |  | ||||||
| 						content, err := ioutil.ReadFile(path) |  | ||||||
| 						if err != nil { |  | ||||||
| 							return err |  | ||||||
| 						} |  | ||||||
|  |  | ||||||
| 						if err := ioutil.WriteFile(path, |  | ||||||
| 							[]byte(generateExpansion(string(content), templateRepo, generateRepo)), |  | ||||||
| 							0644); err != nil { |  | ||||||
| 							return err |  | ||||||
| 						} |  | ||||||
| 						break |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				return nil |  | ||||||
| 			}); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := git.InitRepository(tmpDir, false); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repoPath := repo.RepoPath() |  | ||||||
| 	if stdout, err := git.NewCommand("remote", "add", "origin", repoPath). |  | ||||||
| 		SetDescription(fmt.Sprintf("generateRepoCommit (git remote add): %s to %s", templateRepoPath, tmpDir)). |  | ||||||
| 		RunInDirWithEnv(tmpDir, env); err != nil { |  | ||||||
| 		log.Error("Unable to add %v as remote origin to temporary repo to %s: stdout %s\nError: %v", repo, tmpDir, stdout, err) |  | ||||||
| 		return fmt.Errorf("git remote add: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return initRepoCommit(tmpDir, repo, repo.Owner) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // generateRepository initializes repository from template |  | ||||||
| func generateRepository(e Engine, repo, templateRepo, generateRepo *Repository) (err error) { |  | ||||||
| 	tmpDir, err := ioutil.TempDir(os.TempDir(), "gitea-"+repo.Name) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("Failed to create temp dir for repository %s: %v", repo.RepoPath(), err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	defer func() { |  | ||||||
| 		if err := os.RemoveAll(tmpDir); err != nil { |  | ||||||
| 			log.Error("RemoveAll: %v", err) |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	if err = generateRepoCommit(e, repo, templateRepo, generateRepo, tmpDir); err != nil { |  | ||||||
| 		return fmt.Errorf("generateRepoCommit: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// re-fetch repo |  | ||||||
| 	if repo, err = getRepositoryByID(e, repo.ID); err != nil { |  | ||||||
| 		return fmt.Errorf("getRepositoryByID: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo.DefaultBranch = "master" |  | ||||||
| 	if err = updateRepository(e, repo, false); err != nil { |  | ||||||
| 		return fmt.Errorf("updateRepository: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GenerateRepository generates a repository from a template |  | ||||||
| func GenerateRepository(ctx DBContext, doer, owner *User, templateRepo *Repository, opts GenerateRepoOptions) (_ *Repository, err error) { |  | ||||||
| 	generateRepo := &Repository{ |  | ||||||
| 		OwnerID:       owner.ID, |  | ||||||
| 		Owner:         owner, |  | ||||||
| 		Name:          opts.Name, |  | ||||||
| 		LowerName:     strings.ToLower(opts.Name), |  | ||||||
| 		Description:   opts.Description, |  | ||||||
| 		IsPrivate:     opts.Private, |  | ||||||
| 		IsEmpty:       !opts.GitContent || templateRepo.IsEmpty, |  | ||||||
| 		IsFsckEnabled: templateRepo.IsFsckEnabled, |  | ||||||
| 		TemplateID:    templateRepo.ID, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = createRepository(ctx.e, doer, owner, generateRepo); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repoPath := RepoPath(owner.Name, generateRepo.Name) |  | ||||||
| 	if err = checkInitRepository(repoPath); err != nil { |  | ||||||
| 		return generateRepo, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return generateRepo, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GenerateGitContent generates git content from a template repository |  | ||||||
| func GenerateGitContent(ctx DBContext, templateRepo, generateRepo *Repository) error { |  | ||||||
| 	if err := generateRepository(ctx.e, generateRepo, templateRepo, generateRepo); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := generateRepo.updateSize(ctx.e); err != nil { |  | ||||||
| 		return fmt.Errorf("failed to update size for repository: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := copyLFS(ctx.e, generateRepo, templateRepo); err != nil { |  | ||||||
| 		return fmt.Errorf("failed to copy LFS: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GenerateTopics generates topics from a template repository | // GenerateTopics generates topics from a template repository | ||||||
| func GenerateTopics(ctx DBContext, templateRepo, generateRepo *Repository) error { | func GenerateTopics(ctx DBContext, templateRepo, generateRepo *Repository) error { | ||||||
| 	for _, topic := range templateRepo.Topics { | 	for _, topic := range templateRepo.Topics { | ||||||
| @@ -352,36 +166,3 @@ func GenerateIssueLabels(ctx DBContext, templateRepo, generateRepo *Repository) | |||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func generateExpansion(src string, templateRepo, generateRepo *Repository) string { |  | ||||||
| 	return os.Expand(src, func(key string) string { |  | ||||||
| 		switch key { |  | ||||||
| 		case "REPO_NAME": |  | ||||||
| 			return generateRepo.Name |  | ||||||
| 		case "TEMPLATE_NAME": |  | ||||||
| 			return templateRepo.Name |  | ||||||
| 		case "REPO_DESCRIPTION": |  | ||||||
| 			return generateRepo.Description |  | ||||||
| 		case "TEMPLATE_DESCRIPTION": |  | ||||||
| 			return templateRepo.Description |  | ||||||
| 		case "REPO_OWNER": |  | ||||||
| 			return generateRepo.OwnerName |  | ||||||
| 		case "TEMPLATE_OWNER": |  | ||||||
| 			return templateRepo.OwnerName |  | ||||||
| 		case "REPO_LINK": |  | ||||||
| 			return generateRepo.Link() |  | ||||||
| 		case "TEMPLATE_LINK": |  | ||||||
| 			return templateRepo.Link() |  | ||||||
| 		case "REPO_HTTPS_URL": |  | ||||||
| 			return generateRepo.CloneLink().HTTPS |  | ||||||
| 		case "TEMPLATE_HTTPS_URL": |  | ||||||
| 			return templateRepo.CloneLink().HTTPS |  | ||||||
| 		case "REPO_SSH_URL": |  | ||||||
| 			return generateRepo.CloneLink().SSH |  | ||||||
| 		case "TEMPLATE_SSH_URL": |  | ||||||
| 			return templateRepo.CloneLink().SSH |  | ||||||
| 		default: |  | ||||||
| 			return key |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -133,19 +133,6 @@ func TestGetUserFork(t *testing.T) { | |||||||
| 	assert.Nil(t, repo) | 	assert.Nil(t, repo) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestForkRepository(t *testing.T) { |  | ||||||
| 	assert.NoError(t, PrepareTestDatabase()) |  | ||||||
|  |  | ||||||
| 	// user 13 has already forked repo10 |  | ||||||
| 	user := AssertExistsAndLoadBean(t, &User{ID: 13}).(*User) |  | ||||||
| 	repo := AssertExistsAndLoadBean(t, &Repository{ID: 10}).(*Repository) |  | ||||||
|  |  | ||||||
| 	fork, err := ForkRepository(user, user, repo, "test", "test") |  | ||||||
| 	assert.Nil(t, fork) |  | ||||||
| 	assert.Error(t, err) |  | ||||||
| 	assert.True(t, IsErrForkAlreadyExist(err)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestRepoAPIURL(t *testing.T) { | func TestRepoAPIURL(t *testing.T) { | ||||||
| 	assert.NoError(t, PrepareTestDatabase()) | 	assert.NoError(t, PrepareTestDatabase()) | ||||||
| 	repo := AssertExistsAndLoadBean(t, &Repository{ID: 10}).(*Repository) | 	repo := AssertExistsAndLoadBean(t, &Repository{ID: 10}).(*Repository) | ||||||
|   | |||||||
| @@ -8,8 +8,6 @@ import ( | |||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/modules/log" |  | ||||||
| 	"code.gitea.io/gitea/modules/migrations/base" |  | ||||||
| 	"code.gitea.io/gitea/modules/structs" | 	"code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/timeutil" | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
|  |  | ||||||
| @@ -169,57 +167,16 @@ func FindTasks(opts FindTaskOptions) ([]*Task, error) { | |||||||
| 	return tasks, err | 	return tasks, err | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // CreateTask creates a task on database | ||||||
|  | func CreateTask(task *Task) error { | ||||||
|  | 	return createTask(x, task) | ||||||
|  | } | ||||||
|  |  | ||||||
| func createTask(e Engine, task *Task) error { | func createTask(e Engine, task *Task) error { | ||||||
| 	_, err := e.Insert(task) | 	_, err := e.Insert(task) | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateMigrateTask creates a migrate task |  | ||||||
| func CreateMigrateTask(doer, u *User, opts base.MigrateOptions) (*Task, error) { |  | ||||||
| 	bs, err := json.Marshal(&opts) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var task = Task{ |  | ||||||
| 		DoerID:         doer.ID, |  | ||||||
| 		OwnerID:        u.ID, |  | ||||||
| 		Type:           structs.TaskTypeMigrateRepo, |  | ||||||
| 		Status:         structs.TaskStatusQueue, |  | ||||||
| 		PayloadContent: string(bs), |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := createTask(x, &task); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo, err := CreateRepository(doer, u, CreateRepoOptions{ |  | ||||||
| 		Name:           opts.RepoName, |  | ||||||
| 		Description:    opts.Description, |  | ||||||
| 		OriginalURL:    opts.OriginalURL, |  | ||||||
| 		GitServiceType: opts.GitServiceType, |  | ||||||
| 		IsPrivate:      opts.Private, |  | ||||||
| 		IsMirror:       opts.Mirror, |  | ||||||
| 		Status:         RepositoryBeingMigrated, |  | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		task.EndTime = timeutil.TimeStampNow() |  | ||||||
| 		task.Status = structs.TaskStatusFailed |  | ||||||
| 		err2 := task.UpdateCols("end_time", "status") |  | ||||||
| 		if err2 != nil { |  | ||||||
| 			log.Error("UpdateCols Failed: %v", err2.Error()) |  | ||||||
| 		} |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	task.RepoID = repo.ID |  | ||||||
| 	if err = task.UpdateCols("repo_id"); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &task, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // FinishMigrateTask updates database when migrate task finished | // FinishMigrateTask updates database when migrate task finished | ||||||
| func FinishMigrateTask(task *Task) error { | func FinishMigrateTask(task *Task) error { | ||||||
| 	task.Status = structs.TaskStatusFinished | 	task.Status = structs.TaskStatusFinished | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/migrations/base" | 	"code.gitea.io/gitea/modules/migrations/base" | ||||||
| 	"code.gitea.io/gitea/modules/repository" | 	"code.gitea.io/gitea/modules/repository" | ||||||
|  | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/structs" | 	"code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/timeutil" | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
| @@ -100,7 +101,7 @@ func (g *GiteaLocalUploader) CreateRepo(repo *base.Repository, opts base.Migrate | |||||||
|  |  | ||||||
| 	var r *models.Repository | 	var r *models.Repository | ||||||
| 	if opts.MigrateToRepoID <= 0 { | 	if opts.MigrateToRepoID <= 0 { | ||||||
| 		r, err = models.CreateRepository(g.doer, owner, models.CreateRepoOptions{ | 		r, err = repo_module.CreateRepository(g.doer, owner, models.CreateRepoOptions{ | ||||||
| 			Name:           g.repoName, | 			Name:           g.repoName, | ||||||
| 			Description:    repo.Description, | 			Description:    repo.Description, | ||||||
| 			OriginalURL:    repo.OriginalURL, | 			OriginalURL:    repo.OriginalURL, | ||||||
|   | |||||||
| @@ -458,7 +458,7 @@ func PushUpdate(repo *models.Repository, branch string, opts PushUpdateOptions) | |||||||
| 	} | 	} | ||||||
| 	defer gitRepo.Close() | 	defer gitRepo.Close() | ||||||
|  |  | ||||||
| 	if err = repo.UpdateSize(); err != nil { | 	if err = repo.UpdateSize(models.DefaultDBContext()); err != nil { | ||||||
| 		log.Error("Failed to update size for repository: %v", err) | 		log.Error("Failed to update size for repository: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -498,7 +498,7 @@ func PushUpdates(repo *models.Repository, optsList []*PushUpdateOptions) error { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("OpenRepository: %v", err) | 		return fmt.Errorf("OpenRepository: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err = repo.UpdateSize(); err != nil { | 	if err = repo.UpdateSize(models.DefaultDBContext()); err != nil { | ||||||
| 		log.Error("Failed to update size for repository: %v", err) | 		log.Error("Failed to update size for repository: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										77
									
								
								modules/repository/create.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								modules/repository/create.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | |||||||
|  | // 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 ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	"code.gitea.io/gitea/modules/log" | ||||||
|  | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // CreateRepository creates a repository for the user/organization. | ||||||
|  | func CreateRepository(doer, u *models.User, opts models.CreateRepoOptions) (_ *models.Repository, err error) { | ||||||
|  | 	if !doer.IsAdmin && !u.CanCreateRepo() { | ||||||
|  | 		return nil, models.ErrReachLimitOfRepo{ | ||||||
|  | 			Limit: u.MaxRepoCreation, | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo := &models.Repository{ | ||||||
|  | 		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, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = models.WithTx(func(ctx models.DBContext) error { | ||||||
|  | 		if err = models.CreateRepository(ctx, doer, u, repo); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// No need for init mirror. | ||||||
|  | 		if !opts.IsMirror { | ||||||
|  | 			repoPath := models.RepoPath(u.Name, repo.Name) | ||||||
|  | 			if err = initRepository(ctx, repoPath, u, repo, opts); err != nil { | ||||||
|  | 				if err2 := os.RemoveAll(repoPath); err2 != nil { | ||||||
|  | 					log.Error("initRepository: %v", err) | ||||||
|  | 					return fmt.Errorf( | ||||||
|  | 						"delete repo directory %s/%s failed(2): %v", u.Name, repo.Name, err2) | ||||||
|  | 				} | ||||||
|  | 				return fmt.Errorf("initRepository: %v", err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// Initialize Issue Labels if selected | ||||||
|  | 			if len(opts.IssueLabels) > 0 { | ||||||
|  | 				if err = models.InitalizeLabels(ctx, repo.ID, opts.IssueLabels); err != nil { | ||||||
|  | 					return fmt.Errorf("initalizeLabels: %v", err) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if stdout, err := git.NewCommand("update-server-info"). | ||||||
|  | 				SetDescription(fmt.Sprintf("CreateRepository(git update-server-info): %s", repoPath)). | ||||||
|  | 				RunInDir(repoPath); err != nil { | ||||||
|  | 				log.Error("CreateRepitory(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 | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	return repo, err | ||||||
|  | } | ||||||
							
								
								
									
										145
									
								
								modules/repository/create_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								modules/repository/create_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | |||||||
|  | // 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 ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/structs" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestIncludesAllRepositoriesTeams(t *testing.T) { | ||||||
|  | 	assert.NoError(t, models.PrepareTestDatabase()) | ||||||
|  |  | ||||||
|  | 	testTeamRepositories := func(teamID int64, repoIds []int64) { | ||||||
|  | 		team := models.AssertExistsAndLoadBean(t, &models.Team{ID: teamID}).(*models.Team) | ||||||
|  | 		assert.NoError(t, team.GetRepositories(), "%s: GetRepositories", team.Name) | ||||||
|  | 		assert.Len(t, team.Repos, team.NumRepos, "%s: len repo", team.Name) | ||||||
|  | 		assert.Equal(t, len(repoIds), len(team.Repos), "%s: repo count", team.Name) | ||||||
|  | 		for i, rid := range repoIds { | ||||||
|  | 			if rid > 0 { | ||||||
|  | 				assert.True(t, team.HasRepository(rid), "%s: HasRepository(%d) %d", rid, i) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Get an admin user. | ||||||
|  | 	user, err := models.GetUserByID(1) | ||||||
|  | 	assert.NoError(t, err, "GetUserByID") | ||||||
|  |  | ||||||
|  | 	// Create org. | ||||||
|  | 	org := &models.User{ | ||||||
|  | 		Name:       "All repo", | ||||||
|  | 		IsActive:   true, | ||||||
|  | 		Type:       models.UserTypeOrganization, | ||||||
|  | 		Visibility: structs.VisibleTypePublic, | ||||||
|  | 	} | ||||||
|  | 	assert.NoError(t, models.CreateOrganization(org, user), "CreateOrganization") | ||||||
|  |  | ||||||
|  | 	// Check Owner team. | ||||||
|  | 	ownerTeam, err := org.GetOwnerTeam() | ||||||
|  | 	assert.NoError(t, err, "GetOwnerTeam") | ||||||
|  | 	assert.True(t, ownerTeam.IncludesAllRepositories, "Owner team includes all repositories") | ||||||
|  |  | ||||||
|  | 	// Create repos. | ||||||
|  | 	repoIds := make([]int64, 0) | ||||||
|  | 	for i := 0; i < 3; i++ { | ||||||
|  | 		r, err := CreateRepository(user, org, models.CreateRepoOptions{Name: fmt.Sprintf("repo-%d", i)}) | ||||||
|  | 		assert.NoError(t, err, "CreateRepository %d", i) | ||||||
|  | 		if r != nil { | ||||||
|  | 			repoIds = append(repoIds, r.ID) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	// Get fresh copy of Owner team after creating repos. | ||||||
|  | 	ownerTeam, err = org.GetOwnerTeam() | ||||||
|  | 	assert.NoError(t, err, "GetOwnerTeam") | ||||||
|  |  | ||||||
|  | 	// Create teams and check repositories. | ||||||
|  | 	teams := []*models.Team{ | ||||||
|  | 		ownerTeam, | ||||||
|  | 		{ | ||||||
|  | 			OrgID:                   org.ID, | ||||||
|  | 			Name:                    "team one", | ||||||
|  | 			Authorize:               models.AccessModeRead, | ||||||
|  | 			IncludesAllRepositories: true, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			OrgID:                   org.ID, | ||||||
|  | 			Name:                    "team 2", | ||||||
|  | 			Authorize:               models.AccessModeRead, | ||||||
|  | 			IncludesAllRepositories: false, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			OrgID:                   org.ID, | ||||||
|  | 			Name:                    "team three", | ||||||
|  | 			Authorize:               models.AccessModeWrite, | ||||||
|  | 			IncludesAllRepositories: true, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			OrgID:                   org.ID, | ||||||
|  | 			Name:                    "team 4", | ||||||
|  | 			Authorize:               models.AccessModeWrite, | ||||||
|  | 			IncludesAllRepositories: false, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	teamRepos := [][]int64{ | ||||||
|  | 		repoIds, | ||||||
|  | 		repoIds, | ||||||
|  | 		{}, | ||||||
|  | 		repoIds, | ||||||
|  | 		{}, | ||||||
|  | 	} | ||||||
|  | 	for i, team := range teams { | ||||||
|  | 		if i > 0 { // first team is Owner. | ||||||
|  | 			assert.NoError(t, models.NewTeam(team), "%s: NewTeam", team.Name) | ||||||
|  | 		} | ||||||
|  | 		testTeamRepositories(team.ID, teamRepos[i]) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Update teams and check repositories. | ||||||
|  | 	teams[3].IncludesAllRepositories = false | ||||||
|  | 	teams[4].IncludesAllRepositories = true | ||||||
|  | 	teamRepos[4] = repoIds | ||||||
|  | 	for i, team := range teams { | ||||||
|  | 		assert.NoError(t, models.UpdateTeam(team, false, true), "%s: UpdateTeam", team.Name) | ||||||
|  | 		testTeamRepositories(team.ID, teamRepos[i]) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Create repo and check teams repositories. | ||||||
|  | 	org.Teams = nil // Reset teams to allow their reloading. | ||||||
|  | 	r, err := CreateRepository(user, org, models.CreateRepoOptions{Name: "repo-last"}) | ||||||
|  | 	assert.NoError(t, err, "CreateRepository last") | ||||||
|  | 	if r != nil { | ||||||
|  | 		repoIds = append(repoIds, r.ID) | ||||||
|  | 	} | ||||||
|  | 	teamRepos[0] = repoIds | ||||||
|  | 	teamRepos[1] = repoIds | ||||||
|  | 	teamRepos[4] = repoIds | ||||||
|  | 	for i, team := range teams { | ||||||
|  | 		testTeamRepositories(team.ID, teamRepos[i]) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove repo and check teams repositories. | ||||||
|  | 	assert.NoError(t, models.DeleteRepository(user, org.ID, repoIds[0]), "DeleteRepository") | ||||||
|  | 	teamRepos[0] = repoIds[1:] | ||||||
|  | 	teamRepos[1] = repoIds[1:] | ||||||
|  | 	teamRepos[3] = repoIds[1:3] | ||||||
|  | 	teamRepos[4] = repoIds[1:] | ||||||
|  | 	for i, team := range teams { | ||||||
|  | 		testTeamRepositories(team.ID, teamRepos[i]) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Wipe created items. | ||||||
|  | 	for i, rid := range repoIds { | ||||||
|  | 		if i > 0 { // first repo already deleted. | ||||||
|  | 			assert.NoError(t, models.DeleteRepository(user, org.ID, rid), "DeleteRepository %d", i) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	assert.NoError(t, models.DeleteOrganization(org), "DeleteOrganization") | ||||||
|  | } | ||||||
							
								
								
									
										87
									
								
								modules/repository/fork.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								modules/repository/fork.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | |||||||
|  | // 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 ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	"code.gitea.io/gitea/modules/log" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // ForkRepository forks a repository | ||||||
|  | func ForkRepository(doer, owner *models.User, oldRepo *models.Repository, name, desc string) (_ *models.Repository, err error) { | ||||||
|  | 	forkedRepo, err := oldRepo.GetUserFork(owner.ID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if forkedRepo != nil { | ||||||
|  | 		return nil, models.ErrForkAlreadyExist{ | ||||||
|  | 			Uname:    owner.Name, | ||||||
|  | 			RepoName: oldRepo.FullName(), | ||||||
|  | 			ForkName: forkedRepo.FullName(), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo := &models.Repository{ | ||||||
|  | 		OwnerID:       owner.ID, | ||||||
|  | 		Owner:         owner, | ||||||
|  | 		OwnerName:     owner.Name, | ||||||
|  | 		Name:          name, | ||||||
|  | 		LowerName:     strings.ToLower(name), | ||||||
|  | 		Description:   desc, | ||||||
|  | 		DefaultBranch: oldRepo.DefaultBranch, | ||||||
|  | 		IsPrivate:     oldRepo.IsPrivate, | ||||||
|  | 		IsEmpty:       oldRepo.IsEmpty, | ||||||
|  | 		IsFork:        true, | ||||||
|  | 		ForkID:        oldRepo.ID, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	oldRepoPath := oldRepo.RepoPath() | ||||||
|  |  | ||||||
|  | 	err = models.WithTx(func(ctx models.DBContext) error { | ||||||
|  | 		if err = models.CreateRepository(ctx, doer, owner, repo); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err = models.IncrementRepoForkNum(ctx, oldRepo.ID); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		repoPath := models.RepoPath(owner.Name, repo.Name) | ||||||
|  | 		if stdout, err := git.NewCommand( | ||||||
|  | 			"clone", "--bare", oldRepoPath, repoPath). | ||||||
|  | 			SetDescription(fmt.Sprintf("ForkRepository(git clone): %s to %s", oldRepo.FullName(), repo.FullName())). | ||||||
|  | 			RunInDirTimeout(10*time.Minute, ""); err != nil { | ||||||
|  | 			log.Error("Fork Repository (git clone) Failed for %v (from %v):\nStdout: %s\nError: %v", repo, oldRepo, stdout, err) | ||||||
|  | 			return fmt.Errorf("git clone: %v", err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if stdout, err := git.NewCommand("update-server-info"). | ||||||
|  | 			SetDescription(fmt.Sprintf("ForkRepository(git update-server-info): %s", repo.FullName())). | ||||||
|  | 			RunInDir(repoPath); err != nil { | ||||||
|  | 			log.Error("Fork Repository (git update-server-info) failed for %v:\nStdout: %s\nError: %v", repo, stdout, err) | ||||||
|  | 			return fmt.Errorf("git update-server-info: %v", err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err = models.CreateDelegateHooks(repoPath); err != nil { | ||||||
|  | 			return fmt.Errorf("createDelegateHooks: %v", err) | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ctx := models.DefaultDBContext() | ||||||
|  | 	if err = repo.UpdateSize(ctx); err != nil { | ||||||
|  | 		log.Error("Failed to update size for repository: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return repo, models.CopyLFS(ctx, repo, oldRepo) | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								modules/repository/fork_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								modules/repository/fork_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | // Copyright 2017 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 ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestForkRepository(t *testing.T) { | ||||||
|  | 	assert.NoError(t, models.PrepareTestDatabase()) | ||||||
|  |  | ||||||
|  | 	// user 13 has already forked repo10 | ||||||
|  | 	user := models.AssertExistsAndLoadBean(t, &models.User{ID: 13}).(*models.User) | ||||||
|  | 	repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: 10}).(*models.Repository) | ||||||
|  |  | ||||||
|  | 	fork, err := ForkRepository(user, user, repo, "test", "test") | ||||||
|  | 	assert.Nil(t, fork) | ||||||
|  | 	assert.Error(t, err) | ||||||
|  | 	assert.True(t, models.IsErrForkAlreadyExist(err)) | ||||||
|  | } | ||||||
							
								
								
									
										230
									
								
								modules/repository/generate.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										230
									
								
								modules/repository/generate.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,230 @@ | |||||||
|  | // 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 ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	"code.gitea.io/gitea/modules/log" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func generateExpansion(src string, templateRepo, generateRepo *models.Repository) string { | ||||||
|  | 	return os.Expand(src, func(key string) string { | ||||||
|  | 		switch key { | ||||||
|  | 		case "REPO_NAME": | ||||||
|  | 			return generateRepo.Name | ||||||
|  | 		case "TEMPLATE_NAME": | ||||||
|  | 			return templateRepo.Name | ||||||
|  | 		case "REPO_DESCRIPTION": | ||||||
|  | 			return generateRepo.Description | ||||||
|  | 		case "TEMPLATE_DESCRIPTION": | ||||||
|  | 			return templateRepo.Description | ||||||
|  | 		case "REPO_OWNER": | ||||||
|  | 			return generateRepo.OwnerName | ||||||
|  | 		case "TEMPLATE_OWNER": | ||||||
|  | 			return templateRepo.OwnerName | ||||||
|  | 		case "REPO_LINK": | ||||||
|  | 			return generateRepo.Link() | ||||||
|  | 		case "TEMPLATE_LINK": | ||||||
|  | 			return templateRepo.Link() | ||||||
|  | 		case "REPO_HTTPS_URL": | ||||||
|  | 			return generateRepo.CloneLink().HTTPS | ||||||
|  | 		case "TEMPLATE_HTTPS_URL": | ||||||
|  | 			return templateRepo.CloneLink().HTTPS | ||||||
|  | 		case "REPO_SSH_URL": | ||||||
|  | 			return generateRepo.CloneLink().SSH | ||||||
|  | 		case "TEMPLATE_SSH_URL": | ||||||
|  | 			return templateRepo.CloneLink().SSH | ||||||
|  | 		default: | ||||||
|  | 			return key | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func checkGiteaTemplate(tmpDir string) (*models.GiteaTemplate, error) { | ||||||
|  | 	gtPath := filepath.Join(tmpDir, ".gitea", "template") | ||||||
|  | 	if _, err := os.Stat(gtPath); os.IsNotExist(err) { | ||||||
|  | 		return nil, nil | ||||||
|  | 	} else if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	content, err := ioutil.ReadFile(gtPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	gt := &models.GiteaTemplate{ | ||||||
|  | 		Path:    gtPath, | ||||||
|  | 		Content: content, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return gt, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func generateRepoCommit(repo, templateRepo, generateRepo *models.Repository, tmpDir string) error { | ||||||
|  | 	commitTimeStr := time.Now().Format(time.RFC3339) | ||||||
|  | 	authorSig := repo.Owner.NewGitSig() | ||||||
|  |  | ||||||
|  | 	// Because this may call hooks we should pass in the environment | ||||||
|  | 	env := append(os.Environ(), | ||||||
|  | 		"GIT_AUTHOR_NAME="+authorSig.Name, | ||||||
|  | 		"GIT_AUTHOR_EMAIL="+authorSig.Email, | ||||||
|  | 		"GIT_AUTHOR_DATE="+commitTimeStr, | ||||||
|  | 		"GIT_COMMITTER_NAME="+authorSig.Name, | ||||||
|  | 		"GIT_COMMITTER_EMAIL="+authorSig.Email, | ||||||
|  | 		"GIT_COMMITTER_DATE="+commitTimeStr, | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	// Clone to temporary path and do the init commit. | ||||||
|  | 	templateRepoPath := templateRepo.RepoPath() | ||||||
|  | 	if err := git.Clone(templateRepoPath, tmpDir, git.CloneRepoOptions{ | ||||||
|  | 		Depth: 1, | ||||||
|  | 	}); err != nil { | ||||||
|  | 		return fmt.Errorf("git clone: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := os.RemoveAll(path.Join(tmpDir, ".git")); err != nil { | ||||||
|  | 		return fmt.Errorf("remove git dir: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Variable expansion | ||||||
|  | 	gt, err := checkGiteaTemplate(tmpDir) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("checkGiteaTemplate: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := os.Remove(gt.Path); err != nil { | ||||||
|  | 		return fmt.Errorf("remove .giteatemplate: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Avoid walking tree if there are no globs | ||||||
|  | 	if len(gt.Globs()) > 0 { | ||||||
|  | 		tmpDirSlash := strings.TrimSuffix(filepath.ToSlash(tmpDir), "/") + "/" | ||||||
|  | 		if err := filepath.Walk(tmpDirSlash, func(path string, info os.FileInfo, walkErr error) error { | ||||||
|  | 			if walkErr != nil { | ||||||
|  | 				return walkErr | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if info.IsDir() { | ||||||
|  | 				return nil | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			base := strings.TrimPrefix(filepath.ToSlash(path), tmpDirSlash) | ||||||
|  | 			for _, g := range gt.Globs() { | ||||||
|  | 				if g.Match(base) { | ||||||
|  | 					content, err := ioutil.ReadFile(path) | ||||||
|  | 					if err != nil { | ||||||
|  | 						return err | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					if err := ioutil.WriteFile(path, | ||||||
|  | 						[]byte(generateExpansion(string(content), templateRepo, generateRepo)), | ||||||
|  | 						0644); err != nil { | ||||||
|  | 						return err | ||||||
|  | 					} | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			return nil | ||||||
|  | 		}); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := git.InitRepository(tmpDir, false); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repoPath := repo.RepoPath() | ||||||
|  | 	if stdout, err := git.NewCommand("remote", "add", "origin", repoPath). | ||||||
|  | 		SetDescription(fmt.Sprintf("generateRepoCommit (git remote add): %s to %s", templateRepoPath, tmpDir)). | ||||||
|  | 		RunInDirWithEnv(tmpDir, env); err != nil { | ||||||
|  | 		log.Error("Unable to add %v as remote origin to temporary repo to %s: stdout %s\nError: %v", repo, tmpDir, stdout, err) | ||||||
|  | 		return fmt.Errorf("git remote add: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return initRepoCommit(tmpDir, repo, repo.Owner) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func generateGitContent(ctx models.DBContext, repo, templateRepo, generateRepo *models.Repository) (err error) { | ||||||
|  | 	tmpDir, err := ioutil.TempDir(os.TempDir(), "gitea-"+repo.Name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("Failed to create temp dir for repository %s: %v", repo.RepoPath(), err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	defer func() { | ||||||
|  | 		if err := os.RemoveAll(tmpDir); err != nil { | ||||||
|  | 			log.Error("RemoveAll: %v", err) | ||||||
|  | 		} | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	if err = generateRepoCommit(repo, templateRepo, generateRepo, tmpDir); err != nil { | ||||||
|  | 		return fmt.Errorf("generateRepoCommit: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// re-fetch repo | ||||||
|  | 	if repo, err = models.GetRepositoryByIDCtx(ctx, repo.ID); err != nil { | ||||||
|  | 		return fmt.Errorf("getRepositoryByID: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo.DefaultBranch = "master" | ||||||
|  | 	if err = models.UpdateRepositoryCtx(ctx, repo, false); err != nil { | ||||||
|  | 		return fmt.Errorf("updateRepository: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // GenerateGitContent generates git content from a template repository | ||||||
|  | func GenerateGitContent(ctx models.DBContext, templateRepo, generateRepo *models.Repository) error { | ||||||
|  | 	if err := generateGitContent(ctx, generateRepo, templateRepo, generateRepo); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := generateRepo.UpdateSize(ctx); err != nil { | ||||||
|  | 		return fmt.Errorf("failed to update size for repository: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := models.CopyLFS(ctx, generateRepo, templateRepo); err != nil { | ||||||
|  | 		return fmt.Errorf("failed to copy LFS: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // GenerateRepository generates a repository from a template | ||||||
|  | func GenerateRepository(ctx models.DBContext, doer, owner *models.User, templateRepo *models.Repository, opts models.GenerateRepoOptions) (_ *models.Repository, err error) { | ||||||
|  | 	generateRepo := &models.Repository{ | ||||||
|  | 		OwnerID:       owner.ID, | ||||||
|  | 		Owner:         owner, | ||||||
|  | 		OwnerName:     owner.Name, | ||||||
|  | 		Name:          opts.Name, | ||||||
|  | 		LowerName:     strings.ToLower(opts.Name), | ||||||
|  | 		Description:   opts.Description, | ||||||
|  | 		IsPrivate:     opts.Private, | ||||||
|  | 		IsEmpty:       !opts.GitContent || templateRepo.IsEmpty, | ||||||
|  | 		IsFsckEnabled: templateRepo.IsFsckEnabled, | ||||||
|  | 		TemplateID:    templateRepo.ID, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err = models.CreateRepository(ctx, doer, owner, generateRepo); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repoPath := models.RepoPath(owner.Name, generateRepo.Name) | ||||||
|  | 	if err = checkInitRepository(repoPath); err != nil { | ||||||
|  | 		return generateRepo, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return generateRepo, nil | ||||||
|  | } | ||||||
							
								
								
									
										214
									
								
								modules/repository/init.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										214
									
								
								modules/repository/init.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,214 @@ | |||||||
|  | // 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 ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	"code.gitea.io/gitea/modules/log" | ||||||
|  |  | ||||||
|  | 	"github.com/mcuadros/go-version" | ||||||
|  | 	"github.com/unknwon/com" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func prepareRepoCommit(ctx models.DBContext, repo *models.Repository, tmpDir, repoPath string, opts models.CreateRepoOptions) error { | ||||||
|  | 	commitTimeStr := time.Now().Format(time.RFC3339) | ||||||
|  | 	authorSig := repo.Owner.NewGitSig() | ||||||
|  |  | ||||||
|  | 	// Because this may call hooks we should pass in the environment | ||||||
|  | 	env := append(os.Environ(), | ||||||
|  | 		"GIT_AUTHOR_NAME="+authorSig.Name, | ||||||
|  | 		"GIT_AUTHOR_EMAIL="+authorSig.Email, | ||||||
|  | 		"GIT_AUTHOR_DATE="+commitTimeStr, | ||||||
|  | 		"GIT_COMMITTER_NAME="+authorSig.Name, | ||||||
|  | 		"GIT_COMMITTER_EMAIL="+authorSig.Email, | ||||||
|  | 		"GIT_COMMITTER_DATE="+commitTimeStr, | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	// Clone to temporary path and do the init commit. | ||||||
|  | 	if stdout, err := git.NewCommand("clone", repoPath, tmpDir). | ||||||
|  | 		SetDescription(fmt.Sprintf("prepareRepoCommit (git clone): %s to %s", repoPath, tmpDir)). | ||||||
|  | 		RunInDirWithEnv("", env); err != nil { | ||||||
|  | 		log.Error("Failed to clone from %v into %s: stdout: %s\nError: %v", repo, tmpDir, stdout, err) | ||||||
|  | 		return fmt.Errorf("git clone: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// README | ||||||
|  | 	data, err := models.GetRepoInitFile("readme", opts.Readme) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("GetRepoInitFile[%s]: %v", opts.Readme, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cloneLink := repo.CloneLink() | ||||||
|  | 	match := map[string]string{ | ||||||
|  | 		"Name":           repo.Name, | ||||||
|  | 		"Description":    repo.Description, | ||||||
|  | 		"CloneURL.SSH":   cloneLink.SSH, | ||||||
|  | 		"CloneURL.HTTPS": cloneLink.HTTPS, | ||||||
|  | 	} | ||||||
|  | 	if err = ioutil.WriteFile(filepath.Join(tmpDir, "README.md"), | ||||||
|  | 		[]byte(com.Expand(string(data), match)), 0644); err != nil { | ||||||
|  | 		return fmt.Errorf("write README.md: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// .gitignore | ||||||
|  | 	if len(opts.Gitignores) > 0 { | ||||||
|  | 		var buf bytes.Buffer | ||||||
|  | 		names := strings.Split(opts.Gitignores, ",") | ||||||
|  | 		for _, name := range names { | ||||||
|  | 			data, err = models.GetRepoInitFile("gitignore", name) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("GetRepoInitFile[%s]: %v", name, err) | ||||||
|  | 			} | ||||||
|  | 			buf.WriteString("# ---> " + name + "\n") | ||||||
|  | 			buf.Write(data) | ||||||
|  | 			buf.WriteString("\n") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if buf.Len() > 0 { | ||||||
|  | 			if err = ioutil.WriteFile(filepath.Join(tmpDir, ".gitignore"), buf.Bytes(), 0644); err != nil { | ||||||
|  | 				return fmt.Errorf("write .gitignore: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// LICENSE | ||||||
|  | 	if len(opts.License) > 0 { | ||||||
|  | 		data, err = models.GetRepoInitFile("license", opts.License) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("GetRepoInitFile[%s]: %v", opts.License, err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err = ioutil.WriteFile(filepath.Join(tmpDir, "LICENSE"), data, 0644); err != nil { | ||||||
|  | 			return fmt.Errorf("write LICENSE: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // initRepoCommit temporarily changes with work directory. | ||||||
|  | func initRepoCommit(tmpPath string, repo *models.Repository, u *models.User) (err error) { | ||||||
|  | 	commitTimeStr := time.Now().Format(time.RFC3339) | ||||||
|  |  | ||||||
|  | 	sig := u.NewGitSig() | ||||||
|  | 	// Because this may call hooks we should pass in the environment | ||||||
|  | 	env := append(os.Environ(), | ||||||
|  | 		"GIT_AUTHOR_NAME="+sig.Name, | ||||||
|  | 		"GIT_AUTHOR_EMAIL="+sig.Email, | ||||||
|  | 		"GIT_AUTHOR_DATE="+commitTimeStr, | ||||||
|  | 		"GIT_COMMITTER_NAME="+sig.Name, | ||||||
|  | 		"GIT_COMMITTER_EMAIL="+sig.Email, | ||||||
|  | 		"GIT_COMMITTER_DATE="+commitTimeStr, | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	if stdout, err := git.NewCommand("add", "--all"). | ||||||
|  | 		SetDescription(fmt.Sprintf("initRepoCommit (git add): %s", tmpPath)). | ||||||
|  | 		RunInDir(tmpPath); err != nil { | ||||||
|  | 		log.Error("git add --all failed: Stdout: %s\nError: %v", stdout, err) | ||||||
|  | 		return fmt.Errorf("git add --all: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	binVersion, err := git.BinVersion() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("Unable to get git version: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	args := []string{ | ||||||
|  | 		"commit", fmt.Sprintf("--author='%s <%s>'", sig.Name, sig.Email), | ||||||
|  | 		"-m", "Initial commit", | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if version.Compare(binVersion, "1.7.9", ">=") { | ||||||
|  | 		sign, keyID := models.SignInitialCommit(tmpPath, u) | ||||||
|  | 		if sign { | ||||||
|  | 			args = append(args, "-S"+keyID) | ||||||
|  | 		} else if version.Compare(binVersion, "2.0.0", ">=") { | ||||||
|  | 			args = append(args, "--no-gpg-sign") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if stdout, err := git.NewCommand(args...). | ||||||
|  | 		SetDescription(fmt.Sprintf("initRepoCommit (git commit): %s", tmpPath)). | ||||||
|  | 		RunInDirWithEnv(tmpPath, env); err != nil { | ||||||
|  | 		log.Error("Failed to commit: %v: Stdout: %s\nError: %v", args, stdout, err) | ||||||
|  | 		return fmt.Errorf("git commit: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if stdout, err := git.NewCommand("push", "origin", "master"). | ||||||
|  | 		SetDescription(fmt.Sprintf("initRepoCommit (git push): %s", tmpPath)). | ||||||
|  | 		RunInDirWithEnv(tmpPath, models.InternalPushingEnvironment(u, repo)); err != nil { | ||||||
|  | 		log.Error("Failed to push back to master: Stdout: %s\nError: %v", stdout, err) | ||||||
|  | 		return fmt.Errorf("git push: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func checkInitRepository(repoPath string) (err error) { | ||||||
|  | 	// Somehow the directory could exist. | ||||||
|  | 	if com.IsExist(repoPath) { | ||||||
|  | 		return fmt.Errorf("checkInitRepository: path already exists: %s", repoPath) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Init git bare new repository. | ||||||
|  | 	if err = git.InitRepository(repoPath, true); err != nil { | ||||||
|  | 		return fmt.Errorf("git.InitRepository: %v", err) | ||||||
|  | 	} else if err = models.CreateDelegateHooks(repoPath); err != nil { | ||||||
|  | 		return fmt.Errorf("createDelegateHooks: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // InitRepository initializes README and .gitignore if needed. | ||||||
|  | func initRepository(ctx models.DBContext, repoPath string, u *models.User, repo *models.Repository, opts models.CreateRepoOptions) (err error) { | ||||||
|  | 	if err = checkInitRepository(repoPath); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Initialize repository according to user's choice. | ||||||
|  | 	if opts.AutoInit { | ||||||
|  | 		tmpDir, err := ioutil.TempDir(os.TempDir(), "gitea-"+repo.Name) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("Failed to create temp dir for repository %s: %v", repo.RepoPath(), err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		defer os.RemoveAll(tmpDir) | ||||||
|  |  | ||||||
|  | 		if err = prepareRepoCommit(ctx, repo, tmpDir, repoPath, opts); err != nil { | ||||||
|  | 			return fmt.Errorf("prepareRepoCommit: %v", err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Apply changes and commit. | ||||||
|  | 		if err = initRepoCommit(tmpDir, repo, u); err != nil { | ||||||
|  | 			return fmt.Errorf("initRepoCommit: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Re-fetch the repository from database before updating it (else it would | ||||||
|  | 	// override changes that were done earlier with sql) | ||||||
|  | 	if repo, err = models.GetRepositoryByIDCtx(ctx, repo.ID); err != nil { | ||||||
|  | 		return fmt.Errorf("getRepositoryByID: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !opts.AutoInit { | ||||||
|  | 		repo.IsEmpty = true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo.DefaultBranch = "master" | ||||||
|  | 	if err = models.UpdateRepositoryCtx(ctx, repo, false); err != nil { | ||||||
|  | 		return fmt.Errorf("updateRepository: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @@ -116,7 +116,7 @@ func MigrateRepositoryGitData(doer, u *models.User, repo *models.Repository, opt | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err = repo.UpdateSize(); err != nil { | 	if err = repo.UpdateSize(models.DefaultDBContext()); err != nil { | ||||||
| 		log.Error("Failed to update size for repository: %v", err) | 		log.Error("Failed to update size for repository: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| package task | package task | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
| @@ -12,7 +13,9 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/migrations/base" | 	"code.gitea.io/gitea/modules/migrations/base" | ||||||
| 	"code.gitea.io/gitea/modules/queue" | 	"code.gitea.io/gitea/modules/queue" | ||||||
|  | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| 	"code.gitea.io/gitea/modules/structs" | 	"code.gitea.io/gitea/modules/structs" | ||||||
|  | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // taskQueue is a global queue of tasks | // taskQueue is a global queue of tasks | ||||||
| @@ -52,10 +55,56 @@ func handle(data ...queue.Data) { | |||||||
|  |  | ||||||
| // MigrateRepository add migration repository to task | // MigrateRepository add migration repository to task | ||||||
| func MigrateRepository(doer, u *models.User, opts base.MigrateOptions) error { | func MigrateRepository(doer, u *models.User, opts base.MigrateOptions) error { | ||||||
| 	task, err := models.CreateMigrateTask(doer, u, opts) | 	task, err := CreateMigrateTask(doer, u, opts) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return taskQueue.Push(task) | 	return taskQueue.Push(task) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // CreateMigrateTask creates a migrate task | ||||||
|  | func CreateMigrateTask(doer, u *models.User, opts base.MigrateOptions) (*models.Task, error) { | ||||||
|  | 	bs, err := json.Marshal(&opts) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var task = models.Task{ | ||||||
|  | 		DoerID:         doer.ID, | ||||||
|  | 		OwnerID:        u.ID, | ||||||
|  | 		Type:           structs.TaskTypeMigrateRepo, | ||||||
|  | 		Status:         structs.TaskStatusQueue, | ||||||
|  | 		PayloadContent: string(bs), | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := models.CreateTask(&task); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo, err := repo_module.CreateRepository(doer, u, models.CreateRepoOptions{ | ||||||
|  | 		Name:           opts.RepoName, | ||||||
|  | 		Description:    opts.Description, | ||||||
|  | 		OriginalURL:    opts.OriginalURL, | ||||||
|  | 		GitServiceType: opts.GitServiceType, | ||||||
|  | 		IsPrivate:      opts.Private, | ||||||
|  | 		IsMirror:       opts.Mirror, | ||||||
|  | 		Status:         models.RepositoryBeingMigrated, | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		task.EndTime = timeutil.TimeStampNow() | ||||||
|  | 		task.Status = structs.TaskStatusFailed | ||||||
|  | 		err2 := task.UpdateCols("end_time", "status") | ||||||
|  | 		if err2 != nil { | ||||||
|  | 			log.Error("UpdateCols Failed: %v", err2.Error()) | ||||||
|  | 		} | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	task.RepoID = repo.ID | ||||||
|  | 	if err = task.UpdateCols("repo_id"); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return &task, nil | ||||||
|  | } | ||||||
|   | |||||||
| @@ -22,8 +22,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/migrations" | 	"code.gitea.io/gitea/modules/migrations" | ||||||
| 	"code.gitea.io/gitea/modules/notification" | 	"code.gitea.io/gitea/modules/notification" | ||||||
|  | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/structs" |  | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/util" | 	"code.gitea.io/gitea/modules/util" | ||||||
| 	"code.gitea.io/gitea/modules/validation" | 	"code.gitea.io/gitea/modules/validation" | ||||||
| @@ -451,10 +451,10 @@ func Migrate(ctx *context.APIContext, form auth.MigrateRepoForm) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var gitServiceType = structs.PlainGitService | 	var gitServiceType = api.PlainGitService | ||||||
| 	u, err := url.Parse(remoteAddr) | 	u, err := url.Parse(remoteAddr) | ||||||
| 	if err == nil && strings.EqualFold(u.Host, "github.com") { | 	if err == nil && strings.EqualFold(u.Host, "github.com") { | ||||||
| 		gitServiceType = structs.GithubService | 		gitServiceType = api.GithubService | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var opts = migrations.MigrateOptions{ | 	var opts = migrations.MigrateOptions{ | ||||||
| @@ -483,7 +483,7 @@ func Migrate(ctx *context.APIContext, form auth.MigrateRepoForm) { | |||||||
| 		opts.Releases = false | 		opts.Releases = false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repo, err := models.CreateRepository(ctx.User, ctxUser, models.CreateRepoOptions{ | 	repo, err := repo_module.CreateRepository(ctx.User, ctxUser, models.CreateRepoOptions{ | ||||||
| 		Name:           opts.RepoName, | 		Name:           opts.RepoName, | ||||||
| 		Description:    opts.Description, | 		Description:    opts.Description, | ||||||
| 		OriginalURL:    form.CloneAddr, | 		OriginalURL:    form.CloneAddr, | ||||||
|   | |||||||
| @@ -35,7 +35,7 @@ func InitializeLabels(ctx *context.Context, form auth.InitializeLabelsForm) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := models.InitalizeLabels(ctx.Repo.Repository.ID, form.TemplateName); err != nil { | 	if err := models.InitalizeLabels(models.DefaultDBContext(), ctx.Repo.Repository.ID, form.TemplateName); err != nil { | ||||||
| 		if models.IsErrIssueLabelTemplateLoad(err) { | 		if models.IsErrIssueLabelTemplateLoad(err) { | ||||||
| 			originalErr := err.(models.ErrIssueLabelTemplateLoad).OriginalError | 			originalErr := err.(models.ErrIssueLabelTemplateLoad).OriginalError | ||||||
| 			ctx.Flash.Error(ctx.Tr("repo.issues.label_templates.fail_to_load_file", form.TemplateName, originalErr)) | 			ctx.Flash.Error(ctx.Tr("repo.issues.label_templates.fail_to_load_file", form.TemplateName, originalErr)) | ||||||
|   | |||||||
| @@ -217,7 +217,7 @@ func runSync(m *models.Mirror) ([]*mirrorSyncResult, bool) { | |||||||
| 	} | 	} | ||||||
| 	gitRepo.Close() | 	gitRepo.Close() | ||||||
|  |  | ||||||
| 	if err := m.Repo.UpdateSize(); err != nil { | 	if err := m.Repo.UpdateSize(models.DefaultDBContext()); err != nil { | ||||||
| 		log.Error("Failed to update size for mirror repository: %v", err) | 		log.Error("Failed to update size for mirror repository: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ func TestRelease_MirrorDelete(t *testing.T) { | |||||||
| 		Releases:    false, | 		Releases:    false, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	mirrorRepo, err := models.CreateRepository(user, user, models.CreateRepoOptions{ | 	mirrorRepo, err := repository.CreateRepository(user, user, models.CreateRepoOptions{ | ||||||
| 		Name:        opts.RepoName, | 		Name:        opts.RepoName, | ||||||
| 		Description: opts.Description, | 		Description: opts.Description, | ||||||
| 		IsPrivate:   opts.Private, | 		IsPrivate:   opts.Private, | ||||||
|   | |||||||
| @@ -8,20 +8,21 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/notification" | 	"code.gitea.io/gitea/modules/notification" | ||||||
|  | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // GenerateRepository generates a repository from a template | // GenerateRepository generates a repository from a template | ||||||
| func GenerateRepository(doer, owner *models.User, templateRepo *models.Repository, opts models.GenerateRepoOptions) (_ *models.Repository, err error) { | func GenerateRepository(doer, owner *models.User, templateRepo *models.Repository, opts models.GenerateRepoOptions) (_ *models.Repository, err error) { | ||||||
| 	var generateRepo *models.Repository | 	var generateRepo *models.Repository | ||||||
| 	if err = models.WithTx(func(ctx models.DBContext) error { | 	if err = models.WithTx(func(ctx models.DBContext) error { | ||||||
| 		generateRepo, err = models.GenerateRepository(ctx, doer, owner, templateRepo, opts) | 		generateRepo, err = repo_module.GenerateRepository(ctx, doer, owner, templateRepo, opts) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Git Content | 		// Git Content | ||||||
| 		if opts.GitContent && !templateRepo.IsEmpty { | 		if opts.GitContent && !templateRepo.IsEmpty { | ||||||
| 			if err = models.GenerateGitContent(ctx, templateRepo, generateRepo); err != nil { | 			if err = repo_module.GenerateGitContent(ctx, templateRepo, generateRepo); err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -10,11 +10,12 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/notification" | 	"code.gitea.io/gitea/modules/notification" | ||||||
|  | 	repo_module "code.gitea.io/gitea/modules/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // CreateRepository creates a repository for the user/organization. | // CreateRepository creates a repository for the user/organization. | ||||||
| func CreateRepository(doer, owner *models.User, opts models.CreateRepoOptions) (*models.Repository, error) { | func CreateRepository(doer, owner *models.User, opts models.CreateRepoOptions) (*models.Repository, error) { | ||||||
| 	repo, err := models.CreateRepository(doer, owner, opts) | 	repo, err := repo_module.CreateRepository(doer, owner, opts) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if repo != nil { | 		if repo != nil { | ||||||
| 			if errDelete := models.DeleteRepository(doer, owner.ID, repo.ID); errDelete != nil { | 			if errDelete := models.DeleteRepository(doer, owner.ID, repo.ID); errDelete != nil { | ||||||
| @@ -31,7 +32,7 @@ func CreateRepository(doer, owner *models.User, opts models.CreateRepoOptions) ( | |||||||
|  |  | ||||||
| // ForkRepository forks a repository | // ForkRepository forks a repository | ||||||
| func ForkRepository(doer, u *models.User, oldRepo *models.Repository, name, desc string) (*models.Repository, error) { | func ForkRepository(doer, u *models.User, oldRepo *models.Repository, name, desc string) (*models.Repository, error) { | ||||||
| 	repo, err := models.ForkRepository(doer, u, oldRepo, name, desc) | 	repo, err := repo_module.ForkRepository(doer, u, oldRepo, name, desc) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if repo != nil { | 		if repo != nil { | ||||||
| 			if errDelete := models.DeleteRepository(doer, u.ID, repo.ID); errDelete != nil { | 			if errDelete := models.DeleteRepository(doer, u.ID, repo.ID); errDelete != nil { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user