mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-27 00:23:41 +09:00 
			
		
		
		
	move repository deletion to service layer (#26948)
Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
This commit is contained in:
		| @@ -10,6 +10,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| 	_ "code.gitea.io/gitea/models/activities" | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/auth" | 	_ "code.gitea.io/gitea/models/auth" | ||||||
| 	_ "code.gitea.io/gitea/models/perm/access" | 	_ "code.gitea.io/gitea/models/perm/access" | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -11,6 +11,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/repo" | 	_ "code.gitea.io/gitea/models/repo" | ||||||
| 	_ "code.gitea.io/gitea/models/user" | 	_ "code.gitea.io/gitea/models/user" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| 	_ "code.gitea.io/gitea/models/system" | 	_ "code.gitea.io/gitea/models/system" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
|   | |||||||
| @@ -151,85 +151,6 @@ func removeAllRepositories(ctx context.Context, t *organization.Team) (err error | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // HasRepository returns true if given repository belong to team. |  | ||||||
| func HasRepository(t *organization.Team, repoID int64) bool { |  | ||||||
| 	return organization.HasTeamRepo(db.DefaultContext, t.OrgID, t.ID, repoID) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // removeRepository removes a repository from a team and recalculates access |  | ||||||
| // Note: Repository shall not be removed from team if it includes all repositories (unless the repository is deleted) |  | ||||||
| func removeRepository(ctx context.Context, t *organization.Team, repo *repo_model.Repository, recalculate bool) (err error) { |  | ||||||
| 	e := db.GetEngine(ctx) |  | ||||||
| 	if err = organization.RemoveTeamRepo(ctx, t.ID, repo.ID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	t.NumRepos-- |  | ||||||
| 	if _, err = e.ID(t.ID).Cols("num_repos").Update(t); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Don't need to recalculate when delete a repository from organization. |  | ||||||
| 	if recalculate { |  | ||||||
| 		if err = access_model.RecalculateTeamAccesses(ctx, repo, t.ID); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	teamUsers, err := organization.GetTeamUsersByTeamID(ctx, t.ID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("getTeamUsersByTeamID: %w", err) |  | ||||||
| 	} |  | ||||||
| 	for _, teamUser := range teamUsers { |  | ||||||
| 		has, err := access_model.HasAccess(ctx, teamUser.UID, repo) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} else if has { |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err = repo_model.WatchRepo(ctx, teamUser.UID, repo.ID, false); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Remove all IssueWatches a user has subscribed to in the repositories |  | ||||||
| 		if err := issues_model.RemoveIssueWatchersByRepoID(ctx, teamUser.UID, repo.ID); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // RemoveRepository removes repository from team of organization. |  | ||||||
| // If the team shall include all repositories the request is ignored. |  | ||||||
| func RemoveRepository(t *organization.Team, repoID int64) error { |  | ||||||
| 	if !HasRepository(t, repoID) { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if t.IncludesAllRepositories { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo, err := repo_model.GetRepositoryByID(db.DefaultContext, repoID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	ctx, committer, err := db.TxContext(db.DefaultContext) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer committer.Close() |  | ||||||
|  |  | ||||||
| 	if err = removeRepository(ctx, t, repo, true); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return committer.Commit() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // NewTeam creates a record of new team. | // NewTeam creates a record of new team. | ||||||
| // It's caller's responsibility to assign organization ID. | // It's caller's responsibility to assign organization ID. | ||||||
| func NewTeam(t *organization.Team) (err error) { | func NewTeam(t *organization.Team) (err error) { | ||||||
|   | |||||||
| @@ -51,36 +51,6 @@ func TestTeam_RemoveMember(t *testing.T) { | |||||||
| 	assert.True(t, organization.IsErrLastOrgOwner(err)) | 	assert.True(t, organization.IsErrLastOrgOwner(err)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestTeam_HasRepository(t *testing.T) { |  | ||||||
| 	assert.NoError(t, unittest.PrepareTestDatabase()) |  | ||||||
|  |  | ||||||
| 	test := func(teamID, repoID int64, expected bool) { |  | ||||||
| 		team := unittest.AssertExistsAndLoadBean(t, &organization.Team{ID: teamID}) |  | ||||||
| 		assert.Equal(t, expected, HasRepository(team, repoID)) |  | ||||||
| 	} |  | ||||||
| 	test(1, 1, false) |  | ||||||
| 	test(1, 3, true) |  | ||||||
| 	test(1, 5, true) |  | ||||||
| 	test(1, unittest.NonexistentID, false) |  | ||||||
|  |  | ||||||
| 	test(2, 3, true) |  | ||||||
| 	test(2, 5, false) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestTeam_RemoveRepository(t *testing.T) { |  | ||||||
| 	assert.NoError(t, unittest.PrepareTestDatabase()) |  | ||||||
|  |  | ||||||
| 	testSuccess := func(teamID, repoID int64) { |  | ||||||
| 		team := unittest.AssertExistsAndLoadBean(t, &organization.Team{ID: teamID}) |  | ||||||
| 		assert.NoError(t, RemoveRepository(team, repoID)) |  | ||||||
| 		unittest.AssertNotExistsBean(t, &organization.TeamRepo{TeamID: teamID, RepoID: repoID}) |  | ||||||
| 		unittest.CheckConsistencyFor(t, &organization.Team{ID: teamID}, &repo_model.Repository{ID: repoID}) |  | ||||||
| 	} |  | ||||||
| 	testSuccess(2, 3) |  | ||||||
| 	testSuccess(2, 5) |  | ||||||
| 	testSuccess(1, unittest.NonexistentID) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestIsUsableTeamName(t *testing.T) { | func TestIsUsableTeamName(t *testing.T) { | ||||||
| 	assert.NoError(t, organization.IsUsableTeamName("usable")) | 	assert.NoError(t, organization.IsUsableTeamName("usable")) | ||||||
| 	assert.True(t, db.IsErrNameReserved(organization.IsUsableTeamName("new"))) | 	assert.True(t, db.IsErrNameReserved(organization.IsUsableTeamName("new"))) | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/organization" | 	_ "code.gitea.io/gitea/models/organization" | ||||||
| 	_ "code.gitea.io/gitea/models/repo" | 	_ "code.gitea.io/gitea/models/repo" | ||||||
| 	_ "code.gitea.io/gitea/models/user" | 	_ "code.gitea.io/gitea/models/user" | ||||||
|   | |||||||
| @@ -13,6 +13,8 @@ import ( | |||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/repo" | 	_ "code.gitea.io/gitea/models/repo" | ||||||
| 	_ "code.gitea.io/gitea/models/user" | 	_ "code.gitea.io/gitea/models/user" | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										326
									
								
								models/repo.go
									
									
									
									
									
								
							
							
						
						
									
										326
									
								
								models/repo.go
									
									
									
									
									
								
							| @@ -11,28 +11,15 @@ import ( | |||||||
|  |  | ||||||
| 	_ "image/jpeg" // Needed for jpeg support | 	_ "image/jpeg" // Needed for jpeg support | ||||||
|  |  | ||||||
| 	actions_model "code.gitea.io/gitea/models/actions" |  | ||||||
| 	activities_model "code.gitea.io/gitea/models/activities" |  | ||||||
| 	admin_model "code.gitea.io/gitea/models/admin" |  | ||||||
| 	asymkey_model "code.gitea.io/gitea/models/asymkey" | 	asymkey_model "code.gitea.io/gitea/models/asymkey" | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	git_model "code.gitea.io/gitea/models/git" |  | ||||||
| 	issues_model "code.gitea.io/gitea/models/issues" | 	issues_model "code.gitea.io/gitea/models/issues" | ||||||
| 	"code.gitea.io/gitea/models/organization" |  | ||||||
| 	access_model "code.gitea.io/gitea/models/perm/access" | 	access_model "code.gitea.io/gitea/models/perm/access" | ||||||
| 	project_model "code.gitea.io/gitea/models/project" |  | ||||||
| 	repo_model "code.gitea.io/gitea/models/repo" | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
| 	secret_model "code.gitea.io/gitea/models/secret" |  | ||||||
| 	system_model "code.gitea.io/gitea/models/system" | 	system_model "code.gitea.io/gitea/models/system" | ||||||
| 	"code.gitea.io/gitea/models/unit" | 	"code.gitea.io/gitea/models/unit" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/models/webhook" |  | ||||||
| 	actions_module "code.gitea.io/gitea/modules/actions" |  | ||||||
| 	"code.gitea.io/gitea/modules/lfs" |  | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/storage" |  | ||||||
|  |  | ||||||
| 	"xorm.io/builder" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Init initialize model | // Init initialize model | ||||||
| @@ -43,319 +30,6 @@ func Init(ctx context.Context) error { | |||||||
| 	return system_model.Init(ctx) | 	return system_model.Init(ctx) | ||||||
| } | } | ||||||
|  |  | ||||||
| // DeleteRepository deletes a repository for a user or organization. |  | ||||||
| // make sure if you call this func to close open sessions (sqlite will otherwise get a deadlock) |  | ||||||
| func DeleteRepository(doer *user_model.User, uid, repoID int64) error { |  | ||||||
| 	ctx, committer, err := db.TxContext(db.DefaultContext) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer committer.Close() |  | ||||||
| 	sess := db.GetEngine(ctx) |  | ||||||
|  |  | ||||||
| 	// Query the action tasks of this repo, they will be needed after they have been deleted to remove the logs |  | ||||||
| 	tasks, err := actions_model.FindTasks(ctx, actions_model.FindTaskOptions{RepoID: repoID}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("find actions tasks of repo %v: %w", repoID, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Query the artifacts of this repo, they will be needed after they have been deleted to remove artifacts files in ObjectStorage |  | ||||||
| 	artifacts, err := actions_model.ListArtifactsByRepoID(ctx, repoID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("list actions artifacts of repo %v: %w", repoID, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// In case is a organization. |  | ||||||
| 	org, err := user_model.GetUserByID(ctx, uid) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repo := &repo_model.Repository{OwnerID: uid} |  | ||||||
| 	has, err := sess.ID(repoID).Get(repo) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} else if !has { |  | ||||||
| 		return repo_model.ErrRepoNotExist{ |  | ||||||
| 			ID:        repoID, |  | ||||||
| 			UID:       uid, |  | ||||||
| 			OwnerName: "", |  | ||||||
| 			Name:      "", |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Delete Deploy Keys |  | ||||||
| 	deployKeys, err := asymkey_model.ListDeployKeys(ctx, &asymkey_model.ListDeployKeysOptions{RepoID: repoID}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("listDeployKeys: %w", err) |  | ||||||
| 	} |  | ||||||
| 	needRewriteKeysFile := len(deployKeys) > 0 |  | ||||||
| 	for _, dKey := range deployKeys { |  | ||||||
| 		if err := DeleteDeployKey(ctx, doer, dKey.ID); err != nil { |  | ||||||
| 			return fmt.Errorf("deleteDeployKeys: %w", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if cnt, err := sess.ID(repoID).Delete(&repo_model.Repository{}); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} else if cnt != 1 { |  | ||||||
| 		return repo_model.ErrRepoNotExist{ |  | ||||||
| 			ID:        repoID, |  | ||||||
| 			UID:       uid, |  | ||||||
| 			OwnerName: "", |  | ||||||
| 			Name:      "", |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if org.IsOrganization() { |  | ||||||
| 		teams, err := organization.FindOrgTeams(ctx, org.ID) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		for _, t := range teams { |  | ||||||
| 			if !organization.HasTeamRepo(ctx, t.OrgID, t.ID, repoID) { |  | ||||||
| 				continue |  | ||||||
| 			} else if err = removeRepository(ctx, t, repo, false); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	attachments := make([]*repo_model.Attachment, 0, 20) |  | ||||||
| 	if err = sess.Join("INNER", "`release`", "`release`.id = `attachment`.release_id"). |  | ||||||
| 		Where("`release`.repo_id = ?", repoID). |  | ||||||
| 		Find(&attachments); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	releaseAttachments := make([]string, 0, len(attachments)) |  | ||||||
| 	for i := 0; i < len(attachments); i++ { |  | ||||||
| 		releaseAttachments = append(releaseAttachments, attachments[i].RelativePath()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := db.Exec(ctx, "UPDATE `user` SET num_stars=num_stars-1 WHERE id IN (SELECT `uid` FROM `star` WHERE repo_id = ?)", repo.ID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := db.GetEngine(ctx).In("hook_id", builder.Select("id").From("webhook").Where(builder.Eq{"webhook.repo_id": repo.ID})). |  | ||||||
| 		Delete(&webhook.HookTask{}); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := db.DeleteBeans(ctx, |  | ||||||
| 		&access_model.Access{RepoID: repo.ID}, |  | ||||||
| 		&activities_model.Action{RepoID: repo.ID}, |  | ||||||
| 		&repo_model.Collaboration{RepoID: repoID}, |  | ||||||
| 		&issues_model.Comment{RefRepoID: repoID}, |  | ||||||
| 		&git_model.CommitStatus{RepoID: repoID}, |  | ||||||
| 		&git_model.Branch{RepoID: repoID}, |  | ||||||
| 		&git_model.LFSLock{RepoID: repoID}, |  | ||||||
| 		&repo_model.LanguageStat{RepoID: repoID}, |  | ||||||
| 		&issues_model.Milestone{RepoID: repoID}, |  | ||||||
| 		&repo_model.Mirror{RepoID: repoID}, |  | ||||||
| 		&activities_model.Notification{RepoID: repoID}, |  | ||||||
| 		&git_model.ProtectedBranch{RepoID: repoID}, |  | ||||||
| 		&git_model.ProtectedTag{RepoID: repoID}, |  | ||||||
| 		&repo_model.PushMirror{RepoID: repoID}, |  | ||||||
| 		&repo_model.Release{RepoID: repoID}, |  | ||||||
| 		&repo_model.RepoIndexerStatus{RepoID: repoID}, |  | ||||||
| 		&repo_model.Redirect{RedirectRepoID: repoID}, |  | ||||||
| 		&repo_model.RepoUnit{RepoID: repoID}, |  | ||||||
| 		&repo_model.Star{RepoID: repoID}, |  | ||||||
| 		&admin_model.Task{RepoID: repoID}, |  | ||||||
| 		&repo_model.Watch{RepoID: repoID}, |  | ||||||
| 		&webhook.Webhook{RepoID: repoID}, |  | ||||||
| 		&secret_model.Secret{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionTaskStep{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionTask{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionRunJob{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionRun{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionRunner{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionScheduleSpec{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionSchedule{RepoID: repoID}, |  | ||||||
| 		&actions_model.ActionArtifact{RepoID: repoID}, |  | ||||||
| 	); err != nil { |  | ||||||
| 		return fmt.Errorf("deleteBeans: %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Delete Labels and related objects |  | ||||||
| 	if err := issues_model.DeleteLabelsByRepoID(ctx, repoID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Delete Pulls and related objects |  | ||||||
| 	if err := issues_model.DeletePullsByBaseRepoID(ctx, repoID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Delete Issues and related objects |  | ||||||
| 	var attachmentPaths []string |  | ||||||
| 	if attachmentPaths, err = issues_model.DeleteIssuesByRepoID(ctx, repoID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Delete issue index |  | ||||||
| 	if err := db.DeleteResourceIndex(ctx, "issue_index", repoID); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if repo.IsFork { |  | ||||||
| 		if _, err := db.Exec(ctx, "UPDATE `repository` SET num_forks=num_forks-1 WHERE id=?", repo.ForkID); err != nil { |  | ||||||
| 			return fmt.Errorf("decrease fork count: %w", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := db.Exec(ctx, "UPDATE `user` SET num_repos=num_repos-1 WHERE id=?", uid); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if len(repo.Topics) > 0 { |  | ||||||
| 		if err := repo_model.RemoveTopicsFromRepo(ctx, repo.ID); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := project_model.DeleteProjectByRepoID(ctx, repoID); err != nil { |  | ||||||
| 		return fmt.Errorf("unable to delete projects for repo[%d]: %w", repoID, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove LFS objects |  | ||||||
| 	var lfsObjects []*git_model.LFSMetaObject |  | ||||||
| 	if err = sess.Where("repository_id=?", repoID).Find(&lfsObjects); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lfsPaths := make([]string, 0, len(lfsObjects)) |  | ||||||
| 	for _, v := range lfsObjects { |  | ||||||
| 		count, err := db.CountByBean(ctx, &git_model.LFSMetaObject{Pointer: lfs.Pointer{Oid: v.Oid}}) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		if count > 1 { |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		lfsPaths = append(lfsPaths, v.RelativePath()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := db.DeleteByBean(ctx, &git_model.LFSMetaObject{RepositoryID: repoID}); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove archives |  | ||||||
| 	var archives []*repo_model.RepoArchiver |  | ||||||
| 	if err = sess.Where("repo_id=?", repoID).Find(&archives); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	archivePaths := make([]string, 0, len(archives)) |  | ||||||
| 	for _, v := range archives { |  | ||||||
| 		archivePaths = append(archivePaths, v.RelativePath()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := db.DeleteByBean(ctx, &repo_model.RepoArchiver{RepoID: repoID}); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if repo.NumForks > 0 { |  | ||||||
| 		if _, err = sess.Exec("UPDATE `repository` SET fork_id=0,is_fork=? WHERE fork_id=?", false, repo.ID); err != nil { |  | ||||||
| 			log.Error("reset 'fork_id' and 'is_fork': %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Get all attachments with both issue_id and release_id are zero |  | ||||||
| 	var newAttachments []*repo_model.Attachment |  | ||||||
| 	if err := sess.Where(builder.Eq{ |  | ||||||
| 		"repo_id":    repo.ID, |  | ||||||
| 		"issue_id":   0, |  | ||||||
| 		"release_id": 0, |  | ||||||
| 	}).Find(&newAttachments); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	newAttachmentPaths := make([]string, 0, len(newAttachments)) |  | ||||||
| 	for _, attach := range newAttachments { |  | ||||||
| 		newAttachmentPaths = append(newAttachmentPaths, attach.RelativePath()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if _, err := sess.Where("repo_id=?", repo.ID).Delete(new(repo_model.Attachment)); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err = committer.Commit(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	committer.Close() |  | ||||||
|  |  | ||||||
| 	if needRewriteKeysFile { |  | ||||||
| 		if err := asymkey_model.RewriteAllPublicKeys(); err != nil { |  | ||||||
| 			log.Error("RewriteAllPublicKeys failed: %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// We should always delete the files after the database transaction succeed. If |  | ||||||
| 	// we delete the file but the database rollback, the repository will be broken. |  | ||||||
|  |  | ||||||
| 	// Remove repository files. |  | ||||||
| 	repoPath := repo.RepoPath() |  | ||||||
| 	system_model.RemoveAllWithNotice(db.DefaultContext, "Delete repository files", repoPath) |  | ||||||
|  |  | ||||||
| 	// Remove wiki files |  | ||||||
| 	if repo.HasWiki() { |  | ||||||
| 		system_model.RemoveAllWithNotice(db.DefaultContext, "Delete repository wiki", repo.WikiPath()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove archives |  | ||||||
| 	for _, archive := range archivePaths { |  | ||||||
| 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.RepoArchives, "Delete repo archive file", archive) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove lfs objects |  | ||||||
| 	for _, lfsObj := range lfsPaths { |  | ||||||
| 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.LFS, "Delete orphaned LFS file", lfsObj) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove issue attachment files. |  | ||||||
| 	for _, attachment := range attachmentPaths { |  | ||||||
| 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete issue attachment", attachment) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove release attachment files. |  | ||||||
| 	for _, releaseAttachment := range releaseAttachments { |  | ||||||
| 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete release attachment", releaseAttachment) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Remove attachment with no issue_id and release_id. |  | ||||||
| 	for _, newAttachment := range newAttachmentPaths { |  | ||||||
| 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete issue attachment", newAttachment) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if len(repo.Avatar) > 0 { |  | ||||||
| 		if err := storage.RepoAvatars.Delete(repo.CustomAvatarRelativePath()); err != nil { |  | ||||||
| 			return fmt.Errorf("Failed to remove %s: %w", repo.Avatar, err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Finally, delete action logs after the actions have already been deleted to avoid new log files |  | ||||||
| 	for _, task := range tasks { |  | ||||||
| 		err := actions_module.RemoveLogs(ctx, task.LogInStorage, task.LogFilename) |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Error("remove log file %q: %v", task.LogFilename, err) |  | ||||||
| 			// go on |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// delete actions artifacts in ObjectStorage after the repo have already been deleted |  | ||||||
| 	for _, art := range artifacts { |  | ||||||
| 		if err := storage.ActionsArtifacts.Delete(art.StoragePath); err != nil { |  | ||||||
| 			log.Error("remove artifact file %q: %v", art.StoragePath, err) |  | ||||||
| 			// go on |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type repoChecker struct { | type repoChecker struct { | ||||||
| 	querySQL   func(ctx context.Context) ([]map[string][]byte, error) | 	querySQL   func(ctx context.Context) ([]map[string][]byte, error) | ||||||
| 	correctSQL func(ctx context.Context, id int64) error | 	correctSQL func(ctx context.Context, id int64) error | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" // register table model | 	_ "code.gitea.io/gitea/models" // register table model | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/perm/access" // register table model | 	_ "code.gitea.io/gitea/models/perm/access" // register table model | ||||||
| 	_ "code.gitea.io/gitea/models/repo"        // register table model | 	_ "code.gitea.io/gitea/models/repo"        // register table model | ||||||
| 	_ "code.gitea.io/gitea/models/user"        // register table model | 	_ "code.gitea.io/gitea/models/user"        // register table model | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" // register models | 	_ "code.gitea.io/gitea/models" // register models | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/system" // register models of system | 	_ "code.gitea.io/gitea/models/system" // register models of system | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| 	_ "code.gitea.io/gitea/models/user" | 	_ "code.gitea.io/gitea/models/user" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,8 +15,6 @@ import ( | |||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" // https://discourse.gitea.io/t/testfixtures-could-not-clean-table-access-no-such-table-access/4137/4 |  | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,6 +8,10 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -16,6 +16,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/indexer/code/internal" | 	"code.gitea.io/gitea/modules/indexer/code/internal" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -15,6 +15,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -16,6 +16,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/routers/api/v1/utils" | 	"code.gitea.io/gitea/routers/api/v1/utils" | ||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| 	org_service "code.gitea.io/gitea/services/org" | 	org_service "code.gitea.io/gitea/services/org" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // ListTeams list all the teams of an organization | // ListTeams list all the teams of an organization | ||||||
| @@ -726,7 +727,7 @@ func RemoveTeamRepository(ctx *context.APIContext) { | |||||||
| 		ctx.Error(http.StatusForbidden, "", "Must have admin-level access to the repository") | 		ctx.Error(http.StatusForbidden, "", "Must have admin-level access to the repository") | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	if err := models.RemoveRepository(ctx.Org.Team, repo.ID); err != nil { | 	if err := repo_service.RemoveRepositoryFromTeam(ctx, ctx.Org.Team, repo.ID); err != nil { | ||||||
| 		ctx.Error(http.StatusInternalServerError, "RemoveRepository", err) | 		ctx.Error(http.StatusInternalServerError, "RemoveRepository", err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -170,7 +170,7 @@ func Migrate(ctx *context.APIContext) { | |||||||
| 		opts.Releases = false | 		opts.Releases = false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repo, err := repo_service.CreateRepositoryDirectly(ctx.Doer, repoOwner, repo_service.CreateRepoOptions{ | 	repo, err := repo_service.CreateRepositoryDirectly(ctx, ctx.Doer, repoOwner, repo_service.CreateRepoOptions{ | ||||||
| 		Name:           opts.RepoName, | 		Name:           opts.RepoName, | ||||||
| 		Description:    opts.Description, | 		Description:    opts.Description, | ||||||
| 		OriginalURL:    form.CloneAddr, | 		OriginalURL:    form.CloneAddr, | ||||||
| @@ -200,7 +200,7 @@ func Migrate(ctx *context.APIContext) { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if repo != nil { | 		if repo != nil { | ||||||
| 			if errDelete := models.DeleteRepository(ctx.Doer, repoOwner.ID, repo.ID); errDelete != nil { | 			if errDelete := repo_service.DeleteRepositoryDirectly(ctx, ctx.Doer, repoOwner.ID, repo.ID); errDelete != nil { | ||||||
| 				log.Error("DeleteRepository: %v", errDelete) | 				log.Error("DeleteRepository: %v", errDelete) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -7,11 +7,11 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	"code.gitea.io/gitea/models/organization" | 	"code.gitea.io/gitea/models/organization" | ||||||
| 	"code.gitea.io/gitea/modules/context" | 	"code.gitea.io/gitea/modules/context" | ||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| 	org_service "code.gitea.io/gitea/services/org" | 	org_service "code.gitea.io/gitea/services/org" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // ListTeams list a repository's teams | // ListTeams list a repository's teams | ||||||
| @@ -97,7 +97,7 @@ func IsTeam(ctx *context.APIContext) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if models.HasRepository(team, ctx.Repo.Repository.ID) { | 	if repo_service.HasRepository(team, ctx.Repo.Repository.ID) { | ||||||
| 		apiTeam, err := convert.ToTeam(ctx, team) | 		apiTeam, err := convert.ToTeam(ctx, team) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			ctx.InternalServerError(err) | 			ctx.InternalServerError(err) | ||||||
| @@ -192,7 +192,7 @@ func changeRepoTeam(ctx *context.APIContext, add bool) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repoHasTeam := models.HasRepository(team, ctx.Repo.Repository.ID) | 	repoHasTeam := repo_service.HasRepository(team, ctx.Repo.Repository.ID) | ||||||
| 	var err error | 	var err error | ||||||
| 	if add { | 	if add { | ||||||
| 		if repoHasTeam { | 		if repoHasTeam { | ||||||
| @@ -205,7 +205,7 @@ func changeRepoTeam(ctx *context.APIContext, add bool) { | |||||||
| 			ctx.Error(http.StatusUnprocessableEntity, "notAdded", fmt.Errorf("team '%s' was not added to repo", team.Name)) | 			ctx.Error(http.StatusUnprocessableEntity, "notAdded", fmt.Errorf("team '%s' was not added to repo", team.Name)) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		err = models.RemoveRepository(team, ctx.Repo.Repository.ID) | 		err = repo_service.RemoveRepositoryFromTeam(ctx, team, ctx.Repo.Repository.ID) | ||||||
| 	} | 	} | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		ctx.InternalServerError(err) | 		ctx.InternalServerError(err) | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| 	"code.gitea.io/gitea/services/forms" | 	"code.gitea.io/gitea/services/forms" | ||||||
| 	org_service "code.gitea.io/gitea/services/org" | 	org_service "code.gitea.io/gitea/services/org" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
| @@ -248,7 +249,7 @@ func TeamsRepoAction(ctx *context.Context) { | |||||||
| 		} | 		} | ||||||
| 		err = org_service.TeamAddRepository(ctx.Org.Team, repo) | 		err = org_service.TeamAddRepository(ctx.Org.Team, repo) | ||||||
| 	case "remove": | 	case "remove": | ||||||
| 		err = models.RemoveRepository(ctx.Org.Team, ctx.FormInt64("repoid")) | 		err = repo_service.RemoveRepositoryFromTeam(ctx, ctx.Org.Team, ctx.FormInt64("repoid")) | ||||||
| 	case "addall": | 	case "addall": | ||||||
| 		err = models.AddAllRepositories(ctx.Org.Team) | 		err = models.AddAllRepositories(ctx.Org.Team) | ||||||
| 	case "removeall": | 	case "removeall": | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/routers/utils" | 	"code.gitea.io/gitea/routers/utils" | ||||||
| 	"code.gitea.io/gitea/services/mailer" | 	"code.gitea.io/gitea/services/mailer" | ||||||
| 	org_service "code.gitea.io/gitea/services/org" | 	org_service "code.gitea.io/gitea/services/org" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Collaboration render a repository's collaboration page | // Collaboration render a repository's collaboration page | ||||||
| @@ -196,7 +197,7 @@ func DeleteTeam(ctx *context.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err = models.RemoveRepository(team, ctx.Repo.Repository.ID); err != nil { | 	if err = repo_service.RemoveRepositoryFromTeam(ctx, team, ctx.Repo.Repository.ID); err != nil { | ||||||
| 		ctx.ServerError("team.RemoveRepositorys", err) | 		ctx.ServerError("team.RemoveRepositorys", err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ import ( | |||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	asymkey_model "code.gitea.io/gitea/models/asymkey" | 	asymkey_model "code.gitea.io/gitea/models/asymkey" | ||||||
| 	"code.gitea.io/gitea/models/organization" | 	"code.gitea.io/gitea/models/organization" | ||||||
| 	"code.gitea.io/gitea/models/perm" | 	"code.gitea.io/gitea/models/perm" | ||||||
| @@ -19,6 +18,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/web" | 	"code.gitea.io/gitea/modules/web" | ||||||
| 	"code.gitea.io/gitea/services/forms" | 	"code.gitea.io/gitea/services/forms" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
| @@ -248,7 +248,7 @@ func TestAddTeamPost(t *testing.T) { | |||||||
|  |  | ||||||
| 	AddTeamPost(ctx) | 	AddTeamPost(ctx) | ||||||
|  |  | ||||||
| 	assert.True(t, models.HasRepository(team, re.ID)) | 	assert.True(t, repo_service.HasRepository(team, re.ID)) | ||||||
| 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | ||||||
| 	assert.Empty(t, ctx.Flash.ErrorMsg) | 	assert.Empty(t, ctx.Flash.ErrorMsg) | ||||||
| } | } | ||||||
| @@ -288,7 +288,7 @@ func TestAddTeamPost_NotAllowed(t *testing.T) { | |||||||
|  |  | ||||||
| 	AddTeamPost(ctx) | 	AddTeamPost(ctx) | ||||||
|  |  | ||||||
| 	assert.False(t, models.HasRepository(team, re.ID)) | 	assert.False(t, repo_service.HasRepository(team, re.ID)) | ||||||
| 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | ||||||
| 	assert.NotEmpty(t, ctx.Flash.ErrorMsg) | 	assert.NotEmpty(t, ctx.Flash.ErrorMsg) | ||||||
| } | } | ||||||
| @@ -329,7 +329,7 @@ func TestAddTeamPost_AddTeamTwice(t *testing.T) { | |||||||
| 	AddTeamPost(ctx) | 	AddTeamPost(ctx) | ||||||
|  |  | ||||||
| 	AddTeamPost(ctx) | 	AddTeamPost(ctx) | ||||||
| 	assert.True(t, models.HasRepository(team, re.ID)) | 	assert.True(t, repo_service.HasRepository(team, re.ID)) | ||||||
| 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | 	assert.EqualValues(t, http.StatusSeeOther, ctx.Resp.Status()) | ||||||
| 	assert.NotEmpty(t, ctx.Flash.ErrorMsg) | 	assert.NotEmpty(t, ctx.Flash.ErrorMsg) | ||||||
| } | } | ||||||
| @@ -402,5 +402,5 @@ func TestDeleteTeam(t *testing.T) { | |||||||
|  |  | ||||||
| 	DeleteTeam(ctx) | 	DeleteTeam(ctx) | ||||||
|  |  | ||||||
| 	assert.False(t, models.HasRepository(team, re.ID)) | 	assert.False(t, repo_service.HasRepository(team, re.ID)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,6 +8,9 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -13,6 +13,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  | 	_ "code.gitea.io/gitea/models/activities" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -100,7 +100,7 @@ func (g *GiteaLocalUploader) CreateRepo(repo *base.Repository, opts base.Migrate | |||||||
|  |  | ||||||
| 	var r *repo_model.Repository | 	var r *repo_model.Repository | ||||||
| 	if opts.MigrateToRepoID <= 0 { | 	if opts.MigrateToRepoID <= 0 { | ||||||
| 		r, err = repo_service.CreateRepositoryDirectly(g.doer, owner, repo_service.CreateRepoOptions{ | 		r, err = repo_service.CreateRepositoryDirectly(g.ctx, g.doer, owner, repo_service.CreateRepoOptions{ | ||||||
| 			Name:           g.repoName, | 			Name:           g.repoName, | ||||||
| 			Description:    repo.Description, | 			Description:    repo.Description, | ||||||
| 			OriginalURL:    repo.OriginalURL, | 			OriginalURL:    repo.OriginalURL, | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ import ( | |||||||
| func TeamAddRepository(t *organization.Team, repo *repo_model.Repository) (err error) { | func TeamAddRepository(t *organization.Team, repo *repo_model.Repository) (err error) { | ||||||
| 	if repo.OwnerID != t.OrgID { | 	if repo.OwnerID != t.OrgID { | ||||||
| 		return errors.New("repository does not belong to organization") | 		return errors.New("repository does not belong to organization") | ||||||
| 	} else if models.HasRepository(t, repo.ID) { | 	} else if organization.HasTeamRepo(db.DefaultContext, t.OrgID, t.ID, repo.ID) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -206,7 +206,7 @@ func getOrCreateIndexRepository(ctx context.Context, doer, owner *user_model.Use | |||||||
| 	repo, err := repo_model.GetRepositoryByOwnerAndName(ctx, owner.Name, IndexRepositoryName) | 	repo, err := repo_model.GetRepositoryByOwnerAndName(ctx, owner.Name, IndexRepositoryName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if errors.Is(err, util.ErrNotExist) { | 		if errors.Is(err, util.ErrNotExist) { | ||||||
| 			repo, err = repo_service.CreateRepositoryDirectly(doer, owner, repo_service.CreateRepoOptions{ | 			repo, err = repo_service.CreateRepositoryDirectly(ctx, doer, owner, repo_service.CreateRepoOptions{ | ||||||
| 				Name: IndexRepositoryName, | 				Name: IndexRepositoryName, | ||||||
| 			}) | 			}) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
|   | |||||||
| @@ -9,6 +9,8 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -16,6 +16,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/services/attachment" | 	"code.gitea.io/gitea/services/attachment" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,6 +12,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	"code.gitea.io/gitea/modules/contexttest" | 	"code.gitea.io/gitea/modules/contexttest" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,7 +9,6 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	repo_model "code.gitea.io/gitea/models/repo" | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
| 	system_model "code.gitea.io/gitea/models/system" | 	system_model "code.gitea.io/gitea/models/system" | ||||||
| @@ -165,7 +164,7 @@ func DeleteMissingRepositories(ctx context.Context, doer *user_model.User) error | |||||||
| 		default: | 		default: | ||||||
| 		} | 		} | ||||||
| 		log.Trace("Deleting %d/%d...", repo.OwnerID, repo.ID) | 		log.Trace("Deleting %d/%d...", repo.OwnerID, repo.ID) | ||||||
| 		if err := models.DeleteRepository(doer, repo.OwnerID, repo.ID); err != nil { | 		if err := DeleteRepositoryDirectly(ctx, doer, repo.OwnerID, repo.ID); err != nil { | ||||||
| 			log.Error("Failed to DeleteRepository %-v: Error: %v", repo, err) | 			log.Error("Failed to DeleteRepository %-v: Error: %v", repo, err) | ||||||
| 			if err2 := system_model.CreateRepositoryNotice("Failed to DeleteRepository %s [%d]: Error: %v", repo.FullName(), repo.ID, err); err2 != nil { | 			if err2 := system_model.CreateRepositoryNotice("Failed to DeleteRepository %s [%d]: Error: %v", repo.FullName(), repo.ID, err); err2 != nil { | ||||||
| 				log.Error("CreateRepositoryNotice: %v", err) | 				log.Error("CreateRepositoryNotice: %v", err) | ||||||
|   | |||||||
| @@ -12,7 +12,6 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	repo_model "code.gitea.io/gitea/models/repo" | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| @@ -199,7 +198,7 @@ func initRepository(ctx context.Context, repoPath string, u *user_model.User, re | |||||||
| } | } | ||||||
|  |  | ||||||
| // CreateRepositoryDirectly creates a repository for the user/organization. | // CreateRepositoryDirectly creates a repository for the user/organization. | ||||||
| func CreateRepositoryDirectly(doer, u *user_model.User, opts CreateRepoOptions) (*repo_model.Repository, error) { | func CreateRepositoryDirectly(ctx context.Context, doer, u *user_model.User, opts CreateRepoOptions) (*repo_model.Repository, error) { | ||||||
| 	if !doer.IsAdmin && !u.CanCreateRepo() { | 	if !doer.IsAdmin && !u.CanCreateRepo() { | ||||||
| 		return nil, repo_model.ErrReachLimitOfRepo{ | 		return nil, repo_model.ErrReachLimitOfRepo{ | ||||||
| 			Limit: u.MaxRepoCreation, | 			Limit: u.MaxRepoCreation, | ||||||
| @@ -239,7 +238,7 @@ func CreateRepositoryDirectly(doer, u *user_model.User, opts CreateRepoOptions) | |||||||
|  |  | ||||||
| 	var rollbackRepo *repo_model.Repository | 	var rollbackRepo *repo_model.Repository | ||||||
|  |  | ||||||
| 	if err := db.WithTx(db.DefaultContext, func(ctx context.Context) error { | 	if err := db.WithTx(ctx, func(ctx context.Context) error { | ||||||
| 		if err := repo_module.CreateRepositoryByExample(ctx, doer, u, repo, false, false); err != nil { | 		if err := repo_module.CreateRepositoryByExample(ctx, doer, u, repo, false, false); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| @@ -303,7 +302,7 @@ func CreateRepositoryDirectly(doer, u *user_model.User, opts CreateRepoOptions) | |||||||
| 		return nil | 		return nil | ||||||
| 	}); err != nil { | 	}); err != nil { | ||||||
| 		if rollbackRepo != nil { | 		if rollbackRepo != nil { | ||||||
| 			if errDelete := models.DeleteRepository(doer, rollbackRepo.OwnerID, rollbackRepo.ID); errDelete != nil { | 			if errDelete := DeleteRepositoryDirectly(ctx, doer, rollbackRepo.OwnerID, rollbackRepo.ID); errDelete != nil { | ||||||
| 				log.Error("Rollback deleteRepository: %v", errDelete) | 				log.Error("Rollback deleteRepository: %v", errDelete) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -28,7 +28,7 @@ func TestIncludesAllRepositoriesTeams(t *testing.T) { | |||||||
| 		assert.Len(t, team.Repos, len(repoIds), "%s: repo count", team.Name) | 		assert.Len(t, team.Repos, len(repoIds), "%s: repo count", team.Name) | ||||||
| 		for i, rid := range repoIds { | 		for i, rid := range repoIds { | ||||||
| 			if rid > 0 { | 			if rid > 0 { | ||||||
| 				assert.True(t, models.HasRepository(team, rid), "%s: HasRepository(%d) %d", rid, i) | 				assert.True(t, HasRepository(team, rid), "%s: HasRepository(%d) %d", rid, i) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -54,7 +54,7 @@ func TestIncludesAllRepositoriesTeams(t *testing.T) { | |||||||
| 	// Create repos. | 	// Create repos. | ||||||
| 	repoIds := make([]int64, 0) | 	repoIds := make([]int64, 0) | ||||||
| 	for i := 0; i < 3; i++ { | 	for i := 0; i < 3; i++ { | ||||||
| 		r, err := CreateRepositoryDirectly(user, org.AsUser(), CreateRepoOptions{Name: fmt.Sprintf("repo-%d", i)}) | 		r, err := CreateRepositoryDirectly(db.DefaultContext, user, org.AsUser(), CreateRepoOptions{Name: fmt.Sprintf("repo-%d", i)}) | ||||||
| 		assert.NoError(t, err, "CreateRepository %d", i) | 		assert.NoError(t, err, "CreateRepository %d", i) | ||||||
| 		if r != nil { | 		if r != nil { | ||||||
| 			repoIds = append(repoIds, r.ID) | 			repoIds = append(repoIds, r.ID) | ||||||
| @@ -116,7 +116,7 @@ func TestIncludesAllRepositoriesTeams(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Create repo and check teams repositories. | 	// Create repo and check teams repositories. | ||||||
| 	r, err := CreateRepositoryDirectly(user, org.AsUser(), CreateRepoOptions{Name: "repo-last"}) | 	r, err := CreateRepositoryDirectly(db.DefaultContext, user, org.AsUser(), CreateRepoOptions{Name: "repo-last"}) | ||||||
| 	assert.NoError(t, err, "CreateRepository last") | 	assert.NoError(t, err, "CreateRepository last") | ||||||
| 	if r != nil { | 	if r != nil { | ||||||
| 		repoIds = append(repoIds, r.ID) | 		repoIds = append(repoIds, r.ID) | ||||||
| @@ -129,7 +129,7 @@ func TestIncludesAllRepositoriesTeams(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Remove repo and check teams repositories. | 	// Remove repo and check teams repositories. | ||||||
| 	assert.NoError(t, models.DeleteRepository(user, org.ID, repoIds[0]), "DeleteRepository") | 	assert.NoError(t, DeleteRepositoryDirectly(db.DefaultContext, user, org.ID, repoIds[0]), "DeleteRepository") | ||||||
| 	teamRepos[0] = repoIds[1:] | 	teamRepos[0] = repoIds[1:] | ||||||
| 	teamRepos[1] = repoIds[1:] | 	teamRepos[1] = repoIds[1:] | ||||||
| 	teamRepos[3] = repoIds[1:3] | 	teamRepos[3] = repoIds[1:3] | ||||||
| @@ -141,7 +141,7 @@ func TestIncludesAllRepositoriesTeams(t *testing.T) { | |||||||
| 	// Wipe created items. | 	// Wipe created items. | ||||||
| 	for i, rid := range repoIds { | 	for i, rid := range repoIds { | ||||||
| 		if i > 0 { // first repo already deleted. | 		if i > 0 { // first repo already deleted. | ||||||
| 			assert.NoError(t, models.DeleteRepository(user, org.ID, rid), "DeleteRepository %d", i) | 			assert.NoError(t, DeleteRepositoryDirectly(db.DefaultContext, user, org.ID, rid), "DeleteRepository %d", i) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	assert.NoError(t, organization.DeleteOrganization(db.DefaultContext, org), "DeleteOrganization") | 	assert.NoError(t, organization.DeleteOrganization(db.DefaultContext, org), "DeleteOrganization") | ||||||
|   | |||||||
							
								
								
									
										424
									
								
								services/repository/delete.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										424
									
								
								services/repository/delete.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,424 @@ | |||||||
|  | // Copyright 2023 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  |  | ||||||
|  | package repository | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	actions_model "code.gitea.io/gitea/models/actions" | ||||||
|  | 	activities_model "code.gitea.io/gitea/models/activities" | ||||||
|  | 	admin_model "code.gitea.io/gitea/models/admin" | ||||||
|  | 	asymkey_model "code.gitea.io/gitea/models/asymkey" | ||||||
|  | 	"code.gitea.io/gitea/models/db" | ||||||
|  | 	git_model "code.gitea.io/gitea/models/git" | ||||||
|  | 	issues_model "code.gitea.io/gitea/models/issues" | ||||||
|  | 	"code.gitea.io/gitea/models/organization" | ||||||
|  | 	access_model "code.gitea.io/gitea/models/perm/access" | ||||||
|  | 	project_model "code.gitea.io/gitea/models/project" | ||||||
|  | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
|  | 	secret_model "code.gitea.io/gitea/models/secret" | ||||||
|  | 	system_model "code.gitea.io/gitea/models/system" | ||||||
|  | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  | 	"code.gitea.io/gitea/models/webhook" | ||||||
|  | 	actions_module "code.gitea.io/gitea/modules/actions" | ||||||
|  | 	"code.gitea.io/gitea/modules/lfs" | ||||||
|  | 	"code.gitea.io/gitea/modules/log" | ||||||
|  | 	"code.gitea.io/gitea/modules/storage" | ||||||
|  |  | ||||||
|  | 	"xorm.io/builder" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // DeleteRepository deletes a repository for a user or organization. | ||||||
|  | // make sure if you call this func to close open sessions (sqlite will otherwise get a deadlock) | ||||||
|  | func DeleteRepositoryDirectly(ctx context.Context, doer *user_model.User, uid, repoID int64) error { | ||||||
|  | 	ctx, committer, err := db.TxContext(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer committer.Close() | ||||||
|  | 	sess := db.GetEngine(ctx) | ||||||
|  |  | ||||||
|  | 	// Query the action tasks of this repo, they will be needed after they have been deleted to remove the logs | ||||||
|  | 	tasks, err := actions_model.FindTasks(ctx, actions_model.FindTaskOptions{RepoID: repoID}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("find actions tasks of repo %v: %w", repoID, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Query the artifacts of this repo, they will be needed after they have been deleted to remove artifacts files in ObjectStorage | ||||||
|  | 	artifacts, err := actions_model.ListArtifactsByRepoID(ctx, repoID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("list actions artifacts of repo %v: %w", repoID, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// In case is a organization. | ||||||
|  | 	org, err := user_model.GetUserByID(ctx, uid) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo := &repo_model.Repository{OwnerID: uid} | ||||||
|  | 	has, err := sess.ID(repoID).Get(repo) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} else if !has { | ||||||
|  | 		return repo_model.ErrRepoNotExist{ | ||||||
|  | 			ID:        repoID, | ||||||
|  | 			UID:       uid, | ||||||
|  | 			OwnerName: "", | ||||||
|  | 			Name:      "", | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Delete Deploy Keys | ||||||
|  | 	deployKeys, err := asymkey_model.ListDeployKeys(ctx, &asymkey_model.ListDeployKeysOptions{RepoID: repoID}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("listDeployKeys: %w", err) | ||||||
|  | 	} | ||||||
|  | 	needRewriteKeysFile := len(deployKeys) > 0 | ||||||
|  | 	for _, dKey := range deployKeys { | ||||||
|  | 		if err := models.DeleteDeployKey(ctx, doer, dKey.ID); err != nil { | ||||||
|  | 			return fmt.Errorf("deleteDeployKeys: %w", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if cnt, err := sess.ID(repoID).Delete(&repo_model.Repository{}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} else if cnt != 1 { | ||||||
|  | 		return repo_model.ErrRepoNotExist{ | ||||||
|  | 			ID:        repoID, | ||||||
|  | 			UID:       uid, | ||||||
|  | 			OwnerName: "", | ||||||
|  | 			Name:      "", | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if org.IsOrganization() { | ||||||
|  | 		teams, err := organization.FindOrgTeams(ctx, org.ID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		for _, t := range teams { | ||||||
|  | 			if !organization.HasTeamRepo(ctx, t.OrgID, t.ID, repoID) { | ||||||
|  | 				continue | ||||||
|  | 			} else if err = removeRepositoryFromTeam(ctx, t, repo, false); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	attachments := make([]*repo_model.Attachment, 0, 20) | ||||||
|  | 	if err = sess.Join("INNER", "`release`", "`release`.id = `attachment`.release_id"). | ||||||
|  | 		Where("`release`.repo_id = ?", repoID). | ||||||
|  | 		Find(&attachments); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	releaseAttachments := make([]string, 0, len(attachments)) | ||||||
|  | 	for i := 0; i < len(attachments); i++ { | ||||||
|  | 		releaseAttachments = append(releaseAttachments, attachments[i].RelativePath()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := db.Exec(ctx, "UPDATE `user` SET num_stars=num_stars-1 WHERE id IN (SELECT `uid` FROM `star` WHERE repo_id = ?)", repo.ID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := db.GetEngine(ctx).In("hook_id", builder.Select("id").From("webhook").Where(builder.Eq{"webhook.repo_id": repo.ID})). | ||||||
|  | 		Delete(&webhook.HookTask{}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := db.DeleteBeans(ctx, | ||||||
|  | 		&access_model.Access{RepoID: repo.ID}, | ||||||
|  | 		&activities_model.Action{RepoID: repo.ID}, | ||||||
|  | 		&repo_model.Collaboration{RepoID: repoID}, | ||||||
|  | 		&issues_model.Comment{RefRepoID: repoID}, | ||||||
|  | 		&git_model.CommitStatus{RepoID: repoID}, | ||||||
|  | 		&git_model.Branch{RepoID: repoID}, | ||||||
|  | 		&git_model.LFSLock{RepoID: repoID}, | ||||||
|  | 		&repo_model.LanguageStat{RepoID: repoID}, | ||||||
|  | 		&issues_model.Milestone{RepoID: repoID}, | ||||||
|  | 		&repo_model.Mirror{RepoID: repoID}, | ||||||
|  | 		&activities_model.Notification{RepoID: repoID}, | ||||||
|  | 		&git_model.ProtectedBranch{RepoID: repoID}, | ||||||
|  | 		&git_model.ProtectedTag{RepoID: repoID}, | ||||||
|  | 		&repo_model.PushMirror{RepoID: repoID}, | ||||||
|  | 		&repo_model.Release{RepoID: repoID}, | ||||||
|  | 		&repo_model.RepoIndexerStatus{RepoID: repoID}, | ||||||
|  | 		&repo_model.Redirect{RedirectRepoID: repoID}, | ||||||
|  | 		&repo_model.RepoUnit{RepoID: repoID}, | ||||||
|  | 		&repo_model.Star{RepoID: repoID}, | ||||||
|  | 		&admin_model.Task{RepoID: repoID}, | ||||||
|  | 		&repo_model.Watch{RepoID: repoID}, | ||||||
|  | 		&webhook.Webhook{RepoID: repoID}, | ||||||
|  | 		&secret_model.Secret{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionTaskStep{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionTask{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionRunJob{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionRun{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionRunner{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionScheduleSpec{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionSchedule{RepoID: repoID}, | ||||||
|  | 		&actions_model.ActionArtifact{RepoID: repoID}, | ||||||
|  | 	); err != nil { | ||||||
|  | 		return fmt.Errorf("deleteBeans: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Delete Labels and related objects | ||||||
|  | 	if err := issues_model.DeleteLabelsByRepoID(ctx, repoID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Delete Pulls and related objects | ||||||
|  | 	if err := issues_model.DeletePullsByBaseRepoID(ctx, repoID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Delete Issues and related objects | ||||||
|  | 	var attachmentPaths []string | ||||||
|  | 	if attachmentPaths, err = issues_model.DeleteIssuesByRepoID(ctx, repoID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Delete issue index | ||||||
|  | 	if err := db.DeleteResourceIndex(ctx, "issue_index", repoID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if repo.IsFork { | ||||||
|  | 		if _, err := db.Exec(ctx, "UPDATE `repository` SET num_forks=num_forks-1 WHERE id=?", repo.ForkID); err != nil { | ||||||
|  | 			return fmt.Errorf("decrease fork count: %w", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := db.Exec(ctx, "UPDATE `user` SET num_repos=num_repos-1 WHERE id=?", uid); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(repo.Topics) > 0 { | ||||||
|  | 		if err := repo_model.RemoveTopicsFromRepo(ctx, repo.ID); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err := project_model.DeleteProjectByRepoID(ctx, repoID); err != nil { | ||||||
|  | 		return fmt.Errorf("unable to delete projects for repo[%d]: %w", repoID, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove LFS objects | ||||||
|  | 	var lfsObjects []*git_model.LFSMetaObject | ||||||
|  | 	if err = sess.Where("repository_id=?", repoID).Find(&lfsObjects); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	lfsPaths := make([]string, 0, len(lfsObjects)) | ||||||
|  | 	for _, v := range lfsObjects { | ||||||
|  | 		count, err := db.CountByBean(ctx, &git_model.LFSMetaObject{Pointer: lfs.Pointer{Oid: v.Oid}}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if count > 1 { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		lfsPaths = append(lfsPaths, v.RelativePath()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := db.DeleteByBean(ctx, &git_model.LFSMetaObject{RepositoryID: repoID}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove archives | ||||||
|  | 	var archives []*repo_model.RepoArchiver | ||||||
|  | 	if err = sess.Where("repo_id=?", repoID).Find(&archives); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	archivePaths := make([]string, 0, len(archives)) | ||||||
|  | 	for _, v := range archives { | ||||||
|  | 		archivePaths = append(archivePaths, v.RelativePath()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := db.DeleteByBean(ctx, &repo_model.RepoArchiver{RepoID: repoID}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if repo.NumForks > 0 { | ||||||
|  | 		if _, err = sess.Exec("UPDATE `repository` SET fork_id=0,is_fork=? WHERE fork_id=?", false, repo.ID); err != nil { | ||||||
|  | 			log.Error("reset 'fork_id' and 'is_fork': %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Get all attachments with both issue_id and release_id are zero | ||||||
|  | 	var newAttachments []*repo_model.Attachment | ||||||
|  | 	if err := sess.Where(builder.Eq{ | ||||||
|  | 		"repo_id":    repo.ID, | ||||||
|  | 		"issue_id":   0, | ||||||
|  | 		"release_id": 0, | ||||||
|  | 	}).Find(&newAttachments); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	newAttachmentPaths := make([]string, 0, len(newAttachments)) | ||||||
|  | 	for _, attach := range newAttachments { | ||||||
|  | 		newAttachmentPaths = append(newAttachmentPaths, attach.RelativePath()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if _, err := sess.Where("repo_id=?", repo.ID).Delete(new(repo_model.Attachment)); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err = committer.Commit(); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	committer.Close() | ||||||
|  |  | ||||||
|  | 	if needRewriteKeysFile { | ||||||
|  | 		if err := asymkey_model.RewriteAllPublicKeys(); err != nil { | ||||||
|  | 			log.Error("RewriteAllPublicKeys failed: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// We should always delete the files after the database transaction succeed. If | ||||||
|  | 	// we delete the file but the database rollback, the repository will be broken. | ||||||
|  |  | ||||||
|  | 	// Remove repository files. | ||||||
|  | 	repoPath := repo.RepoPath() | ||||||
|  | 	system_model.RemoveAllWithNotice(db.DefaultContext, "Delete repository files", repoPath) | ||||||
|  |  | ||||||
|  | 	// Remove wiki files | ||||||
|  | 	if repo.HasWiki() { | ||||||
|  | 		system_model.RemoveAllWithNotice(db.DefaultContext, "Delete repository wiki", repo.WikiPath()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove archives | ||||||
|  | 	for _, archive := range archivePaths { | ||||||
|  | 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.RepoArchives, "Delete repo archive file", archive) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove lfs objects | ||||||
|  | 	for _, lfsObj := range lfsPaths { | ||||||
|  | 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.LFS, "Delete orphaned LFS file", lfsObj) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove issue attachment files. | ||||||
|  | 	for _, attachment := range attachmentPaths { | ||||||
|  | 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete issue attachment", attachment) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove release attachment files. | ||||||
|  | 	for _, releaseAttachment := range releaseAttachments { | ||||||
|  | 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete release attachment", releaseAttachment) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Remove attachment with no issue_id and release_id. | ||||||
|  | 	for _, newAttachment := range newAttachmentPaths { | ||||||
|  | 		system_model.RemoveStorageWithNotice(db.DefaultContext, storage.Attachments, "Delete issue attachment", newAttachment) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(repo.Avatar) > 0 { | ||||||
|  | 		if err := storage.RepoAvatars.Delete(repo.CustomAvatarRelativePath()); err != nil { | ||||||
|  | 			return fmt.Errorf("Failed to remove %s: %w", repo.Avatar, err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Finally, delete action logs after the actions have already been deleted to avoid new log files | ||||||
|  | 	for _, task := range tasks { | ||||||
|  | 		err := actions_module.RemoveLogs(ctx, task.LogInStorage, task.LogFilename) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Error("remove log file %q: %v", task.LogFilename, err) | ||||||
|  | 			// go on | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// delete actions artifacts in ObjectStorage after the repo have already been deleted | ||||||
|  | 	for _, art := range artifacts { | ||||||
|  | 		if err := storage.ActionsArtifacts.Delete(art.StoragePath); err != nil { | ||||||
|  | 			log.Error("remove artifact file %q: %v", art.StoragePath, err) | ||||||
|  | 			// go on | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // removeRepositoryFromTeam removes a repository from a team and recalculates access | ||||||
|  | // Note: Repository shall not be removed from team if it includes all repositories (unless the repository is deleted) | ||||||
|  | func removeRepositoryFromTeam(ctx context.Context, t *organization.Team, repo *repo_model.Repository, recalculate bool) (err error) { | ||||||
|  | 	e := db.GetEngine(ctx) | ||||||
|  | 	if err = organization.RemoveTeamRepo(ctx, t.ID, repo.ID); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	t.NumRepos-- | ||||||
|  | 	if _, err = e.ID(t.ID).Cols("num_repos").Update(t); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Don't need to recalculate when delete a repository from organization. | ||||||
|  | 	if recalculate { | ||||||
|  | 		if err = access_model.RecalculateTeamAccesses(ctx, repo, t.ID); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	teamUsers, err := organization.GetTeamUsersByTeamID(ctx, t.ID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("getTeamUsersByTeamID: %w", err) | ||||||
|  | 	} | ||||||
|  | 	for _, teamUser := range teamUsers { | ||||||
|  | 		has, err := access_model.HasAccess(ctx, teamUser.UID, repo) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} else if has { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err = repo_model.WatchRepo(ctx, teamUser.UID, repo.ID, false); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Remove all IssueWatches a user has subscribed to in the repositories | ||||||
|  | 		if err := issues_model.RemoveIssueWatchersByRepoID(ctx, teamUser.UID, repo.ID); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // HasRepository returns true if given repository belong to team. | ||||||
|  | func HasRepository(t *organization.Team, repoID int64) bool { | ||||||
|  | 	return organization.HasTeamRepo(db.DefaultContext, t.OrgID, t.ID, repoID) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // RemoveRepositoryFromTeam removes repository from team of organization. | ||||||
|  | // If the team shall include all repositories the request is ignored. | ||||||
|  | func RemoveRepositoryFromTeam(ctx context.Context, t *organization.Team, repoID int64) error { | ||||||
|  | 	if !HasRepository(t, repoID) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if t.IncludesAllRepositories { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo, err := repo_model.GetRepositoryByID(ctx, repoID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ctx, committer, err := db.TxContext(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer committer.Close() | ||||||
|  |  | ||||||
|  | 	if err = removeRepositoryFromTeam(ctx, t, repo, true); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return committer.Commit() | ||||||
|  | } | ||||||
							
								
								
									
										45
									
								
								services/repository/delete_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								services/repository/delete_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | // Copyright 2017 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  |  | ||||||
|  | package repository | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"code.gitea.io/gitea/models/db" | ||||||
|  | 	"code.gitea.io/gitea/models/organization" | ||||||
|  | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
|  | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestTeam_HasRepository(t *testing.T) { | ||||||
|  | 	assert.NoError(t, unittest.PrepareTestDatabase()) | ||||||
|  |  | ||||||
|  | 	test := func(teamID, repoID int64, expected bool) { | ||||||
|  | 		team := unittest.AssertExistsAndLoadBean(t, &organization.Team{ID: teamID}) | ||||||
|  | 		assert.Equal(t, expected, HasRepository(team, repoID)) | ||||||
|  | 	} | ||||||
|  | 	test(1, 1, false) | ||||||
|  | 	test(1, 3, true) | ||||||
|  | 	test(1, 5, true) | ||||||
|  | 	test(1, unittest.NonexistentID, false) | ||||||
|  |  | ||||||
|  | 	test(2, 3, true) | ||||||
|  | 	test(2, 5, false) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTeam_RemoveRepository(t *testing.T) { | ||||||
|  | 	assert.NoError(t, unittest.PrepareTestDatabase()) | ||||||
|  |  | ||||||
|  | 	testSuccess := func(teamID, repoID int64) { | ||||||
|  | 		team := unittest.AssertExistsAndLoadBean(t, &organization.Team{ID: teamID}) | ||||||
|  | 		assert.NoError(t, RemoveRepositoryFromTeam(db.DefaultContext, team, repoID)) | ||||||
|  | 		unittest.AssertNotExistsBean(t, &organization.TeamRepo{TeamID: teamID, RepoID: repoID}) | ||||||
|  | 		unittest.CheckConsistencyFor(t, &organization.Team{ID: teamID}, &repo_model.Repository{ID: repoID}) | ||||||
|  | 	} | ||||||
|  | 	testSuccess(2, 3) | ||||||
|  | 	testSuccess(2, 5) | ||||||
|  | 	testSuccess(1, unittest.NonexistentID) | ||||||
|  | } | ||||||
| @@ -13,6 +13,8 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	"code.gitea.io/gitea/models/git" | 	"code.gitea.io/gitea/models/git" | ||||||
| 	issues_model "code.gitea.io/gitea/models/issues" | 	issues_model "code.gitea.io/gitea/models/issues" | ||||||
| @@ -41,7 +40,7 @@ type WebSearchResults struct { | |||||||
|  |  | ||||||
| // CreateRepository creates a repository for the user/organization. | // CreateRepository creates a repository for the user/organization. | ||||||
| func CreateRepository(ctx context.Context, doer, owner *user_model.User, opts CreateRepoOptions) (*repo_model.Repository, error) { | func CreateRepository(ctx context.Context, doer, owner *user_model.User, opts CreateRepoOptions) (*repo_model.Repository, error) { | ||||||
| 	repo, err := CreateRepositoryDirectly(doer, owner, opts) | 	repo, err := CreateRepositoryDirectly(ctx, doer, owner, opts) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		// No need to rollback here we should do this in CreateRepository... | 		// No need to rollback here we should do this in CreateRepository... | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @@ -63,7 +62,7 @@ func DeleteRepository(ctx context.Context, doer *user_model.User, repo *repo_mod | |||||||
| 		notify_service.DeleteRepository(ctx, doer, repo) | 		notify_service.DeleteRepository(ctx, doer, repo) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := models.DeleteRepository(doer, repo.OwnerID, repo.ID); err != nil { | 	if err := DeleteRepositoryDirectly(ctx, doer, repo.OwnerID, repo.ID); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	admin_model "code.gitea.io/gitea/models/admin" | 	admin_model "code.gitea.io/gitea/models/admin" | ||||||
|  | 	"code.gitea.io/gitea/models/db" | ||||||
| 	repo_model "code.gitea.io/gitea/models/repo" | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/graceful" | 	"code.gitea.io/gitea/modules/graceful" | ||||||
| @@ -100,7 +101,7 @@ func CreateMigrateTask(doer, u *user_model.User, opts base.MigrateOptions) (*adm | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repo, err := repo_service.CreateRepositoryDirectly(doer, u, repo_service.CreateRepoOptions{ | 	repo, err := repo_service.CreateRepositoryDirectly(db.DefaultContext, doer, u, repo_service.CreateRepoOptions{ | ||||||
| 		Name:           opts.RepoName, | 		Name:           opts.RepoName, | ||||||
| 		Description:    opts.Description, | 		Description:    opts.Description, | ||||||
| 		OriginalURL:    opts.OriginalURL, | 		OriginalURL:    opts.OriginalURL, | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/services/agit" | 	"code.gitea.io/gitea/services/agit" | ||||||
| 	"code.gitea.io/gitea/services/packages" | 	"code.gitea.io/gitea/services/packages" | ||||||
| 	container_service "code.gitea.io/gitea/services/packages/container" | 	container_service "code.gitea.io/gitea/services/packages/container" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // RenameUser renames a user | // RenameUser renames a user | ||||||
| @@ -174,7 +175,7 @@ func DeleteUser(ctx context.Context, u *user_model.User, purge bool) error { | |||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
| 			for _, repo := range repos { | 			for _, repo := range repos { | ||||||
| 				if err := models.DeleteRepository(u, u.ID, repo.ID); err != nil { | 				if err := repo_service.DeleteRepositoryDirectly(ctx, u, u.ID, repo.ID); err != nil { | ||||||
| 					return fmt.Errorf("unable to delete repository %s for %s[%d]. Error: %w", repo.Name, u.Name, u.ID, err) | 					return fmt.Errorf("unable to delete repository %s for %s[%d]. Error: %w", repo.Name, u.Name, u.ID, err) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  |  | ||||||
| 	_ "code.gitea.io/gitea/models" | 	_ "code.gitea.io/gitea/models" | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestMain(m *testing.M) { | func TestMain(m *testing.M) { | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ import ( | |||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
|  |  | ||||||
|  | 	_ "code.gitea.io/gitea/models/actions" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,7 +9,6 @@ import ( | |||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" |  | ||||||
| 	auth_model "code.gitea.io/gitea/models/auth" | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	access_model "code.gitea.io/gitea/models/perm/access" | 	access_model "code.gitea.io/gitea/models/perm/access" | ||||||
| @@ -18,6 +17,7 @@ import ( | |||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
|  | 	repo_service "code.gitea.io/gitea/services/repository" | ||||||
| 	"code.gitea.io/gitea/tests" | 	"code.gitea.io/gitea/tests" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| @@ -541,7 +541,7 @@ func TestAPIRepoTransfer(t *testing.T) { | |||||||
|  |  | ||||||
| 	// cleanup | 	// cleanup | ||||||
| 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: apiRepo.ID}) | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: apiRepo.ID}) | ||||||
| 	_ = models.DeleteRepository(user, repo.OwnerID, repo.ID) | 	_ = repo_service.DeleteRepositoryDirectly(db.DefaultContext, user, repo.OwnerID, repo.ID) | ||||||
| } | } | ||||||
|  |  | ||||||
| func transfer(t *testing.T) *repo_model.Repository { | func transfer(t *testing.T) *repo_model.Repository { | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ func TestMirrorPull(t *testing.T) { | |||||||
| 		Releases:    false, | 		Releases:    false, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	mirrorRepo, err := repo_service.CreateRepositoryDirectly(user, user, repo_service.CreateRepoOptions{ | 	mirrorRepo, err := repo_service.CreateRepositoryDirectly(db.DefaultContext, user, user, repo_service.CreateRepoOptions{ | ||||||
| 		Name:        opts.RepoName, | 		Name:        opts.RepoName, | ||||||
| 		Description: opts.Description, | 		Description: opts.Description, | ||||||
| 		IsPrivate:   opts.Private, | 		IsPrivate:   opts.Private, | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ func testMirrorPush(t *testing.T, u *url.URL) { | |||||||
| 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2}) | 	user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2}) | ||||||
| 	srcRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1}) | 	srcRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1}) | ||||||
|  |  | ||||||
| 	mirrorRepo, err := repo_service.CreateRepositoryDirectly(user, user, repo_service.CreateRepoOptions{ | 	mirrorRepo, err := repo_service.CreateRepositoryDirectly(db.DefaultContext, user, user, repo_service.CreateRepoOptions{ | ||||||
| 		Name: "test-push-mirror", | 		Name: "test-push-mirror", | ||||||
| 	}) | 	}) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user