Merge remote-tracking branch 'upstream/master' into team-grant-all-repos-work
# Conflicts: # models/repo.go
This commit is contained in:
commit
ee52d2dd01
16 changed files with 190 additions and 92 deletions
|
@ -742,6 +742,28 @@ func (repo *Repository) CanUserFork(user *User) (bool, error) {
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CanUserDelete returns true if user could delete the repository
|
||||||
|
func (repo *Repository) CanUserDelete(user *User) (bool, error) {
|
||||||
|
if user.IsAdmin || user.ID == repo.OwnerID {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := repo.GetOwner(); err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if repo.Owner.IsOrganization() {
|
||||||
|
isOwner, err := repo.Owner.IsOwnedBy(user.ID)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
} else if isOwner {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
// CanEnablePulls returns true if repository meets the requirements of accepting pulls.
|
// CanEnablePulls returns true if repository meets the requirements of accepting pulls.
|
||||||
func (repo *Repository) CanEnablePulls() bool {
|
func (repo *Repository) CanEnablePulls() bool {
|
||||||
return !repo.IsMirror && !repo.IsEmpty
|
return !repo.IsMirror && !repo.IsEmpty
|
||||||
|
@ -1437,14 +1459,6 @@ func createRepository(e *xorm.Session, doer, u *User, repo *Repository) (err err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if err := prepareWebhooks(e, repo, HookEventRepository, &api.RepositoryPayload{
|
|
||||||
Action: api.HookRepoCreated,
|
|
||||||
Repository: repo.innerAPIFormat(e, AccessModeOwner, false),
|
|
||||||
Organization: u.APIFormat(),
|
|
||||||
Sender: doer.APIFormat(),
|
|
||||||
}); err != nil {
|
|
||||||
return fmt.Errorf("prepareWebhooks: %v", err)
|
|
||||||
}
|
|
||||||
} else if err = repo.recalculateAccesses(e); err != nil {
|
} else if err = repo.recalculateAccesses(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)
|
||||||
|
@ -1527,11 +1541,6 @@ func CreateRepository(doer, u *User, opts CreateRepoOptions) (_ *Repository, err
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add to hook queue for created repo after session commit.
|
|
||||||
if u.IsOrganization() {
|
|
||||||
go HookQueue.Add(repo.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
return repo, err
|
return repo, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2053,18 +2062,6 @@ func DeleteRepository(doer *User, uid, repoID int64) error {
|
||||||
return fmt.Errorf("Commit: %v", err)
|
return fmt.Errorf("Commit: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if org.IsOrganization() {
|
|
||||||
if err = PrepareWebhooks(repo, HookEventRepository, &api.RepositoryPayload{
|
|
||||||
Action: api.HookRepoDeleted,
|
|
||||||
Repository: repo.APIFormat(AccessModeOwner),
|
|
||||||
Organization: org.APIFormat(),
|
|
||||||
Sender: doer.APIFormat(),
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
go HookQueue.Add(repo.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(repo.Avatar) > 0 {
|
if len(repo.Avatar) > 0 {
|
||||||
avatarPath := repo.CustomAvatarPath()
|
avatarPath := repo.CustomAvatarPath()
|
||||||
if com.IsExist(avatarPath) {
|
if com.IsExist(avatarPath) {
|
||||||
|
@ -2074,7 +2071,6 @@ func DeleteRepository(doer *User, uid, repoID int64) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DeleteRepoFromIndexer(repo)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2530,22 +2526,22 @@ func HasForkedRepo(ownerID, repoID int64) (*Repository, bool) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// ForkRepository forks a repository
|
// ForkRepository forks a repository
|
||||||
func ForkRepository(doer, u *User, oldRepo *Repository, name, desc string) (_ *Repository, err error) {
|
func ForkRepository(doer, owner *User, oldRepo *Repository, name, desc string) (_ *Repository, err error) {
|
||||||
forkedRepo, err := oldRepo.GetUserFork(u.ID)
|
forkedRepo, err := oldRepo.GetUserFork(owner.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if forkedRepo != nil {
|
if forkedRepo != nil {
|
||||||
return nil, ErrForkAlreadyExist{
|
return nil, ErrForkAlreadyExist{
|
||||||
Uname: u.Name,
|
Uname: owner.Name,
|
||||||
RepoName: oldRepo.FullName(),
|
RepoName: oldRepo.FullName(),
|
||||||
ForkName: forkedRepo.FullName(),
|
ForkName: forkedRepo.FullName(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
repo := &Repository{
|
repo := &Repository{
|
||||||
OwnerID: u.ID,
|
OwnerID: owner.ID,
|
||||||
Owner: u,
|
Owner: owner,
|
||||||
Name: name,
|
Name: name,
|
||||||
LowerName: strings.ToLower(name),
|
LowerName: strings.ToLower(name),
|
||||||
Description: desc,
|
Description: desc,
|
||||||
|
@ -2562,7 +2558,7 @@ func ForkRepository(doer, u *User, oldRepo *Repository, name, desc string) (_ *R
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = createRepository(sess, doer, u, repo); err != nil {
|
if err = createRepository(sess, doer, owner, repo); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2570,9 +2566,9 @@ func ForkRepository(doer, u *User, oldRepo *Repository, name, desc string) (_ *R
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
repoPath := RepoPath(u.Name, repo.Name)
|
repoPath := RepoPath(owner.Name, repo.Name)
|
||||||
_, stderr, err := process.GetManager().ExecTimeout(10*time.Minute,
|
_, stderr, err := process.GetManager().ExecTimeout(10*time.Minute,
|
||||||
fmt.Sprintf("ForkRepository(git clone): %s/%s", u.Name, repo.Name),
|
fmt.Sprintf("ForkRepository(git clone): %s/%s", owner.Name, repo.Name),
|
||||||
git.GitExecutable, "clone", "--bare", oldRepo.repoPath(sess), repoPath)
|
git.GitExecutable, "clone", "--bare", oldRepo.repoPath(sess), repoPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("git clone: %v", stderr)
|
return nil, fmt.Errorf("git clone: %v", stderr)
|
||||||
|
@ -2595,24 +2591,6 @@ func ForkRepository(doer, u *User, oldRepo *Repository, name, desc string) (_ *R
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
oldMode, _ := AccessLevel(doer, oldRepo)
|
|
||||||
mode, _ := AccessLevel(doer, repo)
|
|
||||||
|
|
||||||
if err = PrepareWebhooks(oldRepo, HookEventFork, &api.ForkPayload{
|
|
||||||
Forkee: oldRepo.APIFormat(oldMode),
|
|
||||||
Repo: repo.APIFormat(mode),
|
|
||||||
Sender: doer.APIFormat(),
|
|
||||||
}); err != nil {
|
|
||||||
log.Error("PrepareWebhooks [repo_id: %d]: %v", oldRepo.ID, err)
|
|
||||||
} else {
|
|
||||||
go HookQueue.Add(oldRepo.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add to hook queue for created repo after session commit.
|
|
||||||
if u.IsOrganization() {
|
|
||||||
go HookQueue.Add(repo.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = repo.UpdateSize(); err != nil {
|
if err = repo.UpdateSize(); err != nil {
|
||||||
log.Error("Failed to update size for repository: %v", err)
|
log.Error("Failed to update size for repository: %v", err)
|
||||||
}
|
}
|
||||||
|
@ -2621,20 +2599,19 @@ func ForkRepository(doer, u *User, oldRepo *Repository, name, desc string) (_ *R
|
||||||
sess2 := x.NewSession()
|
sess2 := x.NewSession()
|
||||||
defer sess2.Close()
|
defer sess2.Close()
|
||||||
if err = sess2.Begin(); err != nil {
|
if err = sess2.Begin(); err != nil {
|
||||||
return nil, err
|
return repo, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var lfsObjects []*LFSMetaObject
|
var lfsObjects []*LFSMetaObject
|
||||||
|
|
||||||
if err = sess2.Where("repository_id=?", oldRepo.ID).Find(&lfsObjects); err != nil {
|
if err = sess2.Where("repository_id=?", oldRepo.ID).Find(&lfsObjects); err != nil {
|
||||||
return nil, err
|
return repo, err
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, v := range lfsObjects {
|
for _, v := range lfsObjects {
|
||||||
v.ID = 0
|
v.ID = 0
|
||||||
v.RepositoryID = repo.ID
|
v.RepositoryID = repo.ID
|
||||||
if _, err = sess2.Insert(v); err != nil {
|
if _, err = sess2.Insert(v); err != nil {
|
||||||
return nil, err
|
return repo, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -98,6 +98,7 @@ func (r *indexerNotifier) NotifyDeleteComment(doer *models.User, comment *models
|
||||||
|
|
||||||
func (r *indexerNotifier) NotifyDeleteRepository(doer *models.User, repo *models.Repository) {
|
func (r *indexerNotifier) NotifyDeleteRepository(doer *models.User, repo *models.Repository) {
|
||||||
issue_indexer.DeleteRepoIssueIndexer(repo)
|
issue_indexer.DeleteRepoIssueIndexer(repo)
|
||||||
|
models.DeleteRepoFromIndexer(repo)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *indexerNotifier) NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) {
|
func (r *indexerNotifier) NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) {
|
||||||
|
|
|
@ -65,3 +65,67 @@ func (m *webhookNotifier) NotifyIssueClearLabels(doer *models.User, issue *model
|
||||||
go models.HookQueue.Add(issue.RepoID)
|
go models.HookQueue.Add(issue.RepoID)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *webhookNotifier) NotifyForkRepository(doer *models.User, oldRepo, repo *models.Repository) {
|
||||||
|
oldMode, _ := models.AccessLevel(doer, oldRepo)
|
||||||
|
mode, _ := models.AccessLevel(doer, repo)
|
||||||
|
|
||||||
|
// forked webhook
|
||||||
|
if err := models.PrepareWebhooks(oldRepo, models.HookEventFork, &api.ForkPayload{
|
||||||
|
Forkee: oldRepo.APIFormat(oldMode),
|
||||||
|
Repo: repo.APIFormat(mode),
|
||||||
|
Sender: doer.APIFormat(),
|
||||||
|
}); err != nil {
|
||||||
|
log.Error("PrepareWebhooks [repo_id: %d]: %v", oldRepo.ID, err)
|
||||||
|
} else {
|
||||||
|
go models.HookQueue.Add(oldRepo.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
u := repo.MustOwner()
|
||||||
|
|
||||||
|
// Add to hook queue for created repo after session commit.
|
||||||
|
if u.IsOrganization() {
|
||||||
|
if err := models.PrepareWebhooks(repo, models.HookEventRepository, &api.RepositoryPayload{
|
||||||
|
Action: api.HookRepoCreated,
|
||||||
|
Repository: repo.APIFormat(models.AccessModeOwner),
|
||||||
|
Organization: u.APIFormat(),
|
||||||
|
Sender: doer.APIFormat(),
|
||||||
|
}); err != nil {
|
||||||
|
log.Error("PrepareWebhooks [repo_id: %d]: %v", repo.ID, err)
|
||||||
|
} else {
|
||||||
|
go models.HookQueue.Add(repo.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *webhookNotifier) NotifyCreateRepository(doer *models.User, u *models.User, repo *models.Repository) {
|
||||||
|
// Add to hook queue for created repo after session commit.
|
||||||
|
if u.IsOrganization() {
|
||||||
|
if err := models.PrepareWebhooks(repo, models.HookEventRepository, &api.RepositoryPayload{
|
||||||
|
Action: api.HookRepoCreated,
|
||||||
|
Repository: repo.APIFormat(models.AccessModeOwner),
|
||||||
|
Organization: u.APIFormat(),
|
||||||
|
Sender: doer.APIFormat(),
|
||||||
|
}); err != nil {
|
||||||
|
log.Error("PrepareWebhooks [repo_id: %d]: %v", repo.ID, err)
|
||||||
|
} else {
|
||||||
|
go models.HookQueue.Add(repo.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *webhookNotifier) NotifyDeleteRepository(doer *models.User, repo *models.Repository) {
|
||||||
|
u := repo.MustOwner()
|
||||||
|
|
||||||
|
if u.IsOrganization() {
|
||||||
|
if err := models.PrepareWebhooks(repo, models.HookEventRepository, &api.RepositoryPayload{
|
||||||
|
Action: api.HookRepoDeleted,
|
||||||
|
Repository: repo.APIFormat(models.AccessModeOwner),
|
||||||
|
Organization: u.APIFormat(),
|
||||||
|
Sender: doer.APIFormat(),
|
||||||
|
}); err != nil {
|
||||||
|
log.Error("PrepareWebhooks [repo_id: %d]: %v", repo.ID, err)
|
||||||
|
}
|
||||||
|
go models.HookQueue.Add(repo.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -259,6 +259,7 @@ openid_signin_desc=Introduzca su URI OpenID. Por ejemplo: https://anne.me, bob.o
|
||||||
disable_forgot_password_mail=La recuperación de cuentas está desactivada. Por favor, contacte con el administrador del sitio.
|
disable_forgot_password_mail=La recuperación de cuentas está desactivada. Por favor, contacte con el administrador del sitio.
|
||||||
email_domain_blacklisted=No puede registrarse con su correo electrónico.
|
email_domain_blacklisted=No puede registrarse con su correo electrónico.
|
||||||
authorize_application=Autorizar aplicación
|
authorize_application=Autorizar aplicación
|
||||||
|
authorize_redirect_notice=Será redirigido a %s si autoriza esta aplicación.
|
||||||
authorize_application_created_by=Esta aplicación fue creada por %s.
|
authorize_application_created_by=Esta aplicación fue creada por %s.
|
||||||
authorize_application_description=Si concede el acceso, podrá acceder y escribir a toda la información de su cuenta, incluyendo repositorios privado y organizaciones.
|
authorize_application_description=Si concede el acceso, podrá acceder y escribir a toda la información de su cuenta, incluyendo repositorios privado y organizaciones.
|
||||||
authorize_title=¿Autorizar a "%s" a acceder a su cuenta?
|
authorize_title=¿Autorizar a "%s" a acceder a su cuenta?
|
||||||
|
@ -800,6 +801,7 @@ issues.delete_branch_at=`rama eliminada <b>%s</b> %s`
|
||||||
issues.open_tab=%d abiertas
|
issues.open_tab=%d abiertas
|
||||||
issues.close_tab=%d cerradas
|
issues.close_tab=%d cerradas
|
||||||
issues.filter_label=Etiqueta
|
issues.filter_label=Etiqueta
|
||||||
|
issues.filter_label_exclude=`Usa <code>alt</code> + <code>clic/enter</code> para excluir etiquetas`
|
||||||
issues.filter_label_no_select=Todas las etiquetas
|
issues.filter_label_no_select=Todas las etiquetas
|
||||||
issues.filter_milestone=Milestone
|
issues.filter_milestone=Milestone
|
||||||
issues.filter_milestone_no_select=Todos los hitos
|
issues.filter_milestone_no_select=Todos los hitos
|
||||||
|
@ -974,6 +976,7 @@ issues.review.review=Revisar
|
||||||
issues.review.reviewers=Revisores
|
issues.review.reviewers=Revisores
|
||||||
issues.review.show_outdated=Mostrar obsoletos
|
issues.review.show_outdated=Mostrar obsoletos
|
||||||
issues.review.hide_outdated=Ocultar obsoletos
|
issues.review.hide_outdated=Ocultar obsoletos
|
||||||
|
issues.assignee.error=No todos los asignados fueron añadidos debido a un error inesperado.
|
||||||
|
|
||||||
pulls.desc=Activar Pull Requests y revisiones de código.
|
pulls.desc=Activar Pull Requests y revisiones de código.
|
||||||
pulls.new=Nuevo Pull Request
|
pulls.new=Nuevo Pull Request
|
||||||
|
|
|
@ -976,6 +976,7 @@ issues.review.review=Revisão
|
||||||
issues.review.reviewers=Revisões
|
issues.review.reviewers=Revisões
|
||||||
issues.review.show_outdated=Mostrar desatualizado
|
issues.review.show_outdated=Mostrar desatualizado
|
||||||
issues.review.hide_outdated=Ocultar desatualizado
|
issues.review.hide_outdated=Ocultar desatualizado
|
||||||
|
issues.assignee.error=Nem todos os responsáveis foram adicionados devido a um erro inesperado.
|
||||||
|
|
||||||
pulls.desc=Habilitar pull requests e revisões de código.
|
pulls.desc=Habilitar pull requests e revisões de código.
|
||||||
pulls.new=Novo pull request
|
pulls.new=Novo pull request
|
||||||
|
|
|
@ -746,8 +746,8 @@ issues.label_templates.fail_to_load_file=Etiket şablon dosyası yüklemesi baş
|
||||||
issues.add_label_at= %[4]s <div class="ui label has-emoji" style="color: %[1]s\; background-color: %[2]s">%[3]s</div> etiketini eklendi
|
issues.add_label_at= %[4]s <div class="ui label has-emoji" style="color: %[1]s\; background-color: %[2]s">%[3]s</div> etiketini eklendi
|
||||||
issues.remove_label_at=<div class="ui label has-emoji" style="color: %s\; background-color: %s">%s</div> etiketi silindi %s
|
issues.remove_label_at=<div class="ui label has-emoji" style="color: %s\; background-color: %s">%s</div> etiketi silindi %s
|
||||||
issues.add_milestone_at=`%[2]s <b>%[1]s</b> kilometre taşına ekledi`
|
issues.add_milestone_at=`%[2]s <b>%[1]s</b> kilometre taşına ekledi`
|
||||||
issues.change_milestone_at=` <b>%s</b>den<b>%s</b>ye yol taşı düzenlendi %s`
|
issues.change_milestone_at=`%[3]s kilometre taşı <b>%[1]s</b> iken <b>%[2]s</b> olarak değiştirildi`
|
||||||
issues.remove_milestone_at=`bu dosya <b>%s</b> yol taşından kaldırıldı %s`
|
issues.remove_milestone_at=`%[2]s <b>%[1]s</b> kilometre taşından kaldırıldı`
|
||||||
issues.deleted_milestone=`(silindi)`
|
issues.deleted_milestone=`(silindi)`
|
||||||
issues.self_assign_at=`%s kendini atadı`
|
issues.self_assign_at=`%s kendini atadı`
|
||||||
issues.add_assignee_at=`%[2]s <b>%[1]s</b> tarafından atandı`
|
issues.add_assignee_at=`%[2]s <b>%[1]s</b> tarafından atandı`
|
||||||
|
|
|
@ -567,6 +567,7 @@ i.icon.centerlock{top:1.5em}
|
||||||
.repository.view.issue .comment-list .timeline-line:before{display:block;content:"";position:absolute;margin-top:12px;margin-bottom:14px;top:0;bottom:0;left:82px;width:2px;background-color:#f3f3f3;z-index:-1}
|
.repository.view.issue .comment-list .timeline-line:before{display:block;content:"";position:absolute;margin-top:12px;margin-bottom:14px;top:0;bottom:0;left:82px;width:2px;background-color:#f3f3f3;z-index:-1}
|
||||||
.repository.view.issue .comment-list .comment .avatar{width:3em}
|
.repository.view.issue .comment-list .comment .avatar{width:3em}
|
||||||
.repository.view.issue .comment-list .comment .tag{color:#767676;margin-top:3px;padding:2px 5px;font-size:12px;border:1px solid rgba(0,0,0,.1);border-radius:3px}
|
.repository.view.issue .comment-list .comment .tag{color:#767676;margin-top:3px;padding:2px 5px;font-size:12px;border:1px solid rgba(0,0,0,.1);border-radius:3px}
|
||||||
|
.repository.view.issue .comment-list .comment .tag.pending{color:#000;background-color:#fffbb2;margin-left:5px}
|
||||||
.repository.view.issue .comment-list .comment .actions .item{float:left}
|
.repository.view.issue .comment-list .comment .actions .item{float:left}
|
||||||
.repository.view.issue .comment-list .comment .actions .item.tag{margin-right:5px}
|
.repository.view.issue .comment-list .comment .actions .item.tag{margin-right:5px}
|
||||||
.repository.view.issue .comment-list .comment .actions .item.action{margin-top:6px;margin-left:10px}
|
.repository.view.issue .comment-list .comment .actions .item.action{margin-top:6px;margin-left:10px}
|
||||||
|
|
|
@ -738,6 +738,11 @@
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
border: 1px solid rgba(0, 0, 0, 0.1);
|
border: 1px solid rgba(0, 0, 0, 0.1);
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
|
&.pending {
|
||||||
|
color: black;
|
||||||
|
background-color: #fffbb2;
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions {
|
.actions {
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/routers"
|
"code.gitea.io/gitea/routers"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -38,7 +39,7 @@ func DeleteRepo(ctx *context.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := models.DeleteRepository(ctx.User, repo.MustOwner().ID, repo.ID); err != nil {
|
if err := repo_service.DeleteRepository(ctx.User, repo); err != nil {
|
||||||
ctx.ServerError("DeleteRepository", err)
|
ctx.ServerError("DeleteRepository", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import (
|
||||||
"code.gitea.io/gitea/models"
|
"code.gitea.io/gitea/models"
|
||||||
"code.gitea.io/gitea/modules/context"
|
"code.gitea.io/gitea/modules/context"
|
||||||
api "code.gitea.io/gitea/modules/structs"
|
api "code.gitea.io/gitea/modules/structs"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ListForks list a repository's forks
|
// ListForks list a repository's forks
|
||||||
|
@ -97,10 +98,12 @@ func CreateFork(ctx *context.APIContext, form api.CreateForkOption) {
|
||||||
}
|
}
|
||||||
forker = org
|
forker = org
|
||||||
}
|
}
|
||||||
fork, err := models.ForkRepository(ctx.User, forker, repo, repo.Name, repo.Description)
|
|
||||||
|
fork, err := repo_service.ForkRepository(ctx.User, forker, repo, repo.Name, repo.Description)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.Error(500, "ForkRepository", err)
|
ctx.Error(500, "ForkRepository", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.JSON(202, fork.APIFormat(models.AccessModeOwner))
|
ctx.JSON(202, fork.APIFormat(models.AccessModeOwner))
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,7 @@ import (
|
||||||
"code.gitea.io/gitea/modules/validation"
|
"code.gitea.io/gitea/modules/validation"
|
||||||
"code.gitea.io/gitea/routers/api/v1/convert"
|
"code.gitea.io/gitea/routers/api/v1/convert"
|
||||||
mirror_service "code.gitea.io/gitea/services/mirror"
|
mirror_service "code.gitea.io/gitea/services/mirror"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
)
|
)
|
||||||
|
|
||||||
var searchOrderByMap = map[string]map[string]models.SearchOrderBy{
|
var searchOrderByMap = map[string]map[string]models.SearchOrderBy{
|
||||||
|
@ -207,7 +208,7 @@ func CreateUserRepo(ctx *context.APIContext, owner *models.User, opt api.CreateR
|
||||||
if opt.AutoInit && opt.Readme == "" {
|
if opt.AutoInit && opt.Readme == "" {
|
||||||
opt.Readme = "Default"
|
opt.Readme = "Default"
|
||||||
}
|
}
|
||||||
repo, err := models.CreateRepository(ctx.User, owner, models.CreateRepoOptions{
|
repo, err := repo_service.CreateRepository(ctx.User, owner, models.CreateRepoOptions{
|
||||||
Name: opt.Name,
|
Name: opt.Name,
|
||||||
Description: opt.Description,
|
Description: opt.Description,
|
||||||
IssueLabels: opt.IssueLabels,
|
IssueLabels: opt.IssueLabels,
|
||||||
|
@ -224,18 +225,11 @@ func CreateUserRepo(ctx *context.APIContext, owner *models.User, opt api.CreateR
|
||||||
models.IsErrNamePatternNotAllowed(err) {
|
models.IsErrNamePatternNotAllowed(err) {
|
||||||
ctx.Error(422, "", err)
|
ctx.Error(422, "", err)
|
||||||
} else {
|
} else {
|
||||||
if repo != nil {
|
|
||||||
if err = models.DeleteRepository(ctx.User, ctx.User.ID, repo.ID); err != nil {
|
|
||||||
log.Error("DeleteRepository: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ctx.Error(500, "CreateRepository", err)
|
ctx.Error(500, "CreateRepository", err)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.NotifyCreateRepository(ctx.User, owner, repo)
|
|
||||||
|
|
||||||
ctx.JSON(201, repo.APIFormat(models.AccessModeOwner))
|
ctx.JSON(201, repo.APIFormat(models.AccessModeOwner))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -433,7 +427,7 @@ func Migrate(ctx *context.APIContext, form auth.MigrateRepoForm) {
|
||||||
|
|
||||||
repo, err := migrations.MigrateRepository(ctx.User, ctxUser.Name, opts)
|
repo, err := migrations.MigrateRepository(ctx.User, ctxUser.Name, opts)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
notification.NotifyCreateRepository(ctx.User, ctxUser, repo)
|
notification.NotifyMigrateRepository(ctx.User, ctxUser, repo)
|
||||||
|
|
||||||
log.Trace("Repository migrated: %s/%s", ctxUser.Name, form.RepoName)
|
log.Trace("Repository migrated: %s/%s", ctxUser.Name, form.RepoName)
|
||||||
ctx.JSON(201, repo.APIFormat(models.AccessModeAdmin))
|
ctx.JSON(201, repo.APIFormat(models.AccessModeAdmin))
|
||||||
|
@ -876,18 +870,16 @@ func Delete(ctx *context.APIContext) {
|
||||||
owner := ctx.Repo.Owner
|
owner := ctx.Repo.Owner
|
||||||
repo := ctx.Repo.Repository
|
repo := ctx.Repo.Repository
|
||||||
|
|
||||||
if owner.IsOrganization() && !ctx.User.IsAdmin {
|
canDelete, err := repo.CanUserDelete(ctx.User)
|
||||||
isOwner, err := owner.IsOwnedBy(ctx.User.ID)
|
if err != nil {
|
||||||
if err != nil {
|
ctx.Error(500, "CanUserDelete", err)
|
||||||
ctx.Error(500, "IsOwnedBy", err)
|
return
|
||||||
return
|
} else if !canDelete {
|
||||||
} else if !isOwner {
|
ctx.Error(403, "", "Given user is not owner of organization.")
|
||||||
ctx.Error(403, "", "Given user is not owner of organization.")
|
return
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := models.DeleteRepository(ctx.User, owner.ID, repo.ID); err != nil {
|
if err := repo_service.DeleteRepository(ctx.User, repo); err != nil {
|
||||||
ctx.Error(500, "DeleteRepository", err)
|
ctx.Error(500, "DeleteRepository", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ import (
|
||||||
"code.gitea.io/gitea/services/gitdiff"
|
"code.gitea.io/gitea/services/gitdiff"
|
||||||
issue_service "code.gitea.io/gitea/services/issue"
|
issue_service "code.gitea.io/gitea/services/issue"
|
||||||
pull_service "code.gitea.io/gitea/services/pull"
|
pull_service "code.gitea.io/gitea/services/pull"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
|
|
||||||
"github.com/unknwon/com"
|
"github.com/unknwon/com"
|
||||||
)
|
)
|
||||||
|
@ -209,7 +210,7 @@ func ForkPost(ctx *context.Context, form auth.CreateRepoForm) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
repo, err := models.ForkRepository(ctx.User, ctxUser, forkRepo, form.RepoName, form.Description)
|
repo, err := repo_service.ForkRepository(ctx.User, ctxUser, forkRepo, form.RepoName, form.Description)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.Data["Err_RepoName"] = true
|
ctx.Data["Err_RepoName"] = true
|
||||||
switch {
|
switch {
|
||||||
|
|
|
@ -17,10 +17,10 @@ import (
|
||||||
"code.gitea.io/gitea/modules/git"
|
"code.gitea.io/gitea/modules/git"
|
||||||
"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/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/task"
|
"code.gitea.io/gitea/modules/task"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
|
|
||||||
"github.com/unknwon/com"
|
"github.com/unknwon/com"
|
||||||
)
|
)
|
||||||
|
@ -170,7 +170,7 @@ func CreatePost(ctx *context.Context, form auth.CreateRepoForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
repo, err := models.CreateRepository(ctx.User, ctxUser, models.CreateRepoOptions{
|
repo, err := repo_service.CreateRepository(ctx.User, ctxUser, models.CreateRepoOptions{
|
||||||
Name: form.RepoName,
|
Name: form.RepoName,
|
||||||
Description: form.Description,
|
Description: form.Description,
|
||||||
Gitignores: form.Gitignores,
|
Gitignores: form.Gitignores,
|
||||||
|
@ -181,19 +181,11 @@ func CreatePost(ctx *context.Context, form auth.CreateRepoForm) {
|
||||||
AutoInit: form.AutoInit,
|
AutoInit: form.AutoInit,
|
||||||
})
|
})
|
||||||
if err == nil {
|
if err == nil {
|
||||||
notification.NotifyCreateRepository(ctx.User, ctxUser, repo)
|
|
||||||
|
|
||||||
log.Trace("Repository created [%d]: %s/%s", repo.ID, ctxUser.Name, repo.Name)
|
log.Trace("Repository created [%d]: %s/%s", repo.ID, ctxUser.Name, repo.Name)
|
||||||
ctx.Redirect(setting.AppSubURL + "/" + ctxUser.Name + "/" + repo.Name)
|
ctx.Redirect(setting.AppSubURL + "/" + ctxUser.Name + "/" + repo.Name)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if repo != nil {
|
|
||||||
if errDelete := models.DeleteRepository(ctx.User, ctxUser.ID, repo.ID); errDelete != nil {
|
|
||||||
log.Error("DeleteRepository: %v", errDelete)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
handleCreateError(ctx, ctxUser, err, "CreatePost", tplCreate, &form)
|
handleCreateError(ctx, ctxUser, err, "CreatePost", tplCreate, &form)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,6 +26,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"
|
||||||
mirror_service "code.gitea.io/gitea/services/mirror"
|
mirror_service "code.gitea.io/gitea/services/mirror"
|
||||||
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
|
|
||||||
"github.com/unknwon/com"
|
"github.com/unknwon/com"
|
||||||
"mvdan.cc/xurls/v2"
|
"mvdan.cc/xurls/v2"
|
||||||
|
@ -407,7 +408,7 @@ func SettingsPost(ctx *context.Context, form auth.RepoSettingForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := models.DeleteRepository(ctx.User, ctx.Repo.Owner.ID, repo.ID); err != nil {
|
if err := repo_service.DeleteRepository(ctx.User, ctx.Repo.Repository); err != nil {
|
||||||
ctx.ServerError("DeleteRepository", err)
|
ctx.ServerError("DeleteRepository", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
56
services/repository/repository.go
Normal file
56
services/repository/repository.go
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
// 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 (
|
||||||
|
"code.gitea.io/gitea/models"
|
||||||
|
"code.gitea.io/gitea/modules/log"
|
||||||
|
"code.gitea.io/gitea/modules/notification"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CreateRepository creates a repository for the user/organization.
|
||||||
|
func CreateRepository(doer, owner *models.User, opts models.CreateRepoOptions) (*models.Repository, error) {
|
||||||
|
repo, err := models.CreateRepository(doer, owner, opts)
|
||||||
|
if err != nil {
|
||||||
|
if repo != nil {
|
||||||
|
if errDelete := models.DeleteRepository(doer, owner.ID, repo.ID); errDelete != nil {
|
||||||
|
log.Error("Rollback deleteRepository: %v", errDelete)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
notification.NotifyCreateRepository(doer, owner, repo)
|
||||||
|
|
||||||
|
return repo, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ForkRepository forks a repository
|
||||||
|
func ForkRepository(doer, u *models.User, oldRepo *models.Repository, name, desc string) (*models.Repository, error) {
|
||||||
|
repo, err := models.ForkRepository(doer, u, oldRepo, name, desc)
|
||||||
|
if err != nil {
|
||||||
|
if repo != nil {
|
||||||
|
if errDelete := models.DeleteRepository(doer, u.ID, repo.ID); errDelete != nil {
|
||||||
|
log.Error("Rollback deleteRepository: %v", errDelete)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
notification.NotifyForkRepository(doer, oldRepo, repo)
|
||||||
|
|
||||||
|
return repo, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteRepository deletes a repository for a user or organization.
|
||||||
|
func DeleteRepository(doer *models.User, repo *models.Repository) error {
|
||||||
|
if err := models.DeleteRepository(doer, repo.OwnerID, repo.ID); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
notification.NotifyDeleteRepository(doer, repo)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -11,7 +11,7 @@
|
||||||
<div class="ui right actions">
|
<div class="ui right actions">
|
||||||
{{if and .Review}}
|
{{if and .Review}}
|
||||||
{{if eq .Review.Type 0}}
|
{{if eq .Review.Type 0}}
|
||||||
<div class="item tag">
|
<div class="item tag pending">
|
||||||
{{$.root.i18n.Tr "repo.issues.review.pending"}}
|
{{$.root.i18n.Tr "repo.issues.review.pending"}}
|
||||||
</div>
|
</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
|
|
Loading…
Reference in a new issue