Sync releases table with tags on push and for mirrors (#2459)
* Sync releases table with tags on push and for mirrors * Code style fixes * Fix api to return only releases * Optimize release creation and update Minimize posibility of race conditions * Fix release lower tag name updating * handle tag reference update by addionally comparing commit id
This commit is contained in:
		
					parent
					
						
							
								8b6236d67b
							
						
					
				
			
			
				commit
				
					
						7a0297819d
					
				
			
		
					 10 changed files with 370 additions and 122 deletions
				
			
		| 
						 | 
					@ -132,6 +132,8 @@ var migrations = []Migration{
 | 
				
			||||||
	NewMigration("migrate protected branch struct", migrateProtectedBranchStruct),
 | 
						NewMigration("migrate protected branch struct", migrateProtectedBranchStruct),
 | 
				
			||||||
	// v41 -> v42
 | 
						// v41 -> v42
 | 
				
			||||||
	NewMigration("add default value to user prohibit_login", addDefaultValueToUserProhibitLogin),
 | 
						NewMigration("add default value to user prohibit_login", addDefaultValueToUserProhibitLogin),
 | 
				
			||||||
 | 
						// v42 -> v43
 | 
				
			||||||
 | 
						NewMigration("add tags to releases and sync existing repositories", releaseAddColumnIsTagAndSyncTags),
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Migrate database to current version
 | 
					// Migrate database to current version
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										57
									
								
								models/migrations/v42.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								models/migrations/v42.go
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,57 @@
 | 
				
			||||||
 | 
					// Copyright 2017 The Gitea Authors. All rights reserved.
 | 
				
			||||||
 | 
					// Use of this source code is governed by a MIT-style
 | 
				
			||||||
 | 
					// license that can be found in the LICENSE file.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package migrations
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/git"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/modules/log"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/go-xorm/xorm"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ReleaseV39 describes the added field for Release
 | 
				
			||||||
 | 
					type ReleaseV39 struct {
 | 
				
			||||||
 | 
						IsTag bool `xorm:"NOT NULL DEFAULT false"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TableName will be invoked by XORM to customrize the table name
 | 
				
			||||||
 | 
					func (*ReleaseV39) TableName() string {
 | 
				
			||||||
 | 
						return "release"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func releaseAddColumnIsTagAndSyncTags(x *xorm.Engine) error {
 | 
				
			||||||
 | 
						if err := x.Sync2(new(ReleaseV39)); err != nil {
 | 
				
			||||||
 | 
							return fmt.Errorf("Sync2: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// For the sake of SQLite3, we can't use x.Iterate here.
 | 
				
			||||||
 | 
						offset := 0
 | 
				
			||||||
 | 
						pageSize := 20
 | 
				
			||||||
 | 
						for {
 | 
				
			||||||
 | 
							repos := make([]*models.Repository, 0, pageSize)
 | 
				
			||||||
 | 
							if err := x.Table("repository").Asc("id").Limit(pageSize, offset).Find(&repos); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("select repos [offset: %d]: %v", offset, err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							for _, repo := range repos {
 | 
				
			||||||
 | 
								gitRepo, err := git.OpenRepository(repo.RepoPath())
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									log.Warn("OpenRepository: %v", err)
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if err = models.SyncReleasesWithTags(repo, gitRepo); err != nil {
 | 
				
			||||||
 | 
									log.Warn("SyncReleasesWithTags: %v", err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if len(repos) < pageSize {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							offset += pageSize
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -34,7 +34,8 @@ type Release struct {
 | 
				
			||||||
	NumCommitsBehind int64  `xorm:"-"`
 | 
						NumCommitsBehind int64  `xorm:"-"`
 | 
				
			||||||
	Note             string `xorm:"TEXT"`
 | 
						Note             string `xorm:"TEXT"`
 | 
				
			||||||
	IsDraft          bool   `xorm:"NOT NULL DEFAULT false"`
 | 
						IsDraft          bool   `xorm:"NOT NULL DEFAULT false"`
 | 
				
			||||||
	IsPrerelease     bool
 | 
						IsPrerelease     bool   `xorm:"NOT NULL DEFAULT false"`
 | 
				
			||||||
 | 
						IsTag            bool   `xorm:"NOT NULL DEFAULT false"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	Attachments []*Attachment `xorm:"-"`
 | 
						Attachments []*Attachment `xorm:"-"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -139,17 +140,18 @@ func createTag(gitRepo *git.Repository, rel *Release) error {
 | 
				
			||||||
				}
 | 
									}
 | 
				
			||||||
				return err
 | 
									return err
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		} else {
 | 
								rel.LowerTagName = strings.ToLower(rel.TagName)
 | 
				
			||||||
			commit, err := gitRepo.GetTagCommit(rel.TagName)
 | 
							}
 | 
				
			||||||
			if err != nil {
 | 
							commit, err := gitRepo.GetTagCommit(rel.TagName)
 | 
				
			||||||
				return fmt.Errorf("GetTagCommit: %v", err)
 | 
							if err != nil {
 | 
				
			||||||
			}
 | 
								return fmt.Errorf("GetTagCommit: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			rel.Sha1 = commit.ID.String()
 | 
							rel.Sha1 = commit.ID.String()
 | 
				
			||||||
			rel.NumCommits, err = commit.CommitsCount()
 | 
							rel.CreatedUnix = commit.Author.When.Unix()
 | 
				
			||||||
			if err != nil {
 | 
							rel.NumCommits, err = commit.CommitsCount()
 | 
				
			||||||
				return fmt.Errorf("CommitsCount: %v", err)
 | 
							if err != nil {
 | 
				
			||||||
			}
 | 
								return fmt.Errorf("CommitsCount: %v", err)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return nil
 | 
						return nil
 | 
				
			||||||
| 
						 | 
					@ -236,6 +238,7 @@ func GetReleaseByID(id int64) (*Release, error) {
 | 
				
			||||||
// FindReleasesOptions describes the conditions to Find releases
 | 
					// FindReleasesOptions describes the conditions to Find releases
 | 
				
			||||||
type FindReleasesOptions struct {
 | 
					type FindReleasesOptions struct {
 | 
				
			||||||
	IncludeDrafts bool
 | 
						IncludeDrafts bool
 | 
				
			||||||
 | 
						IncludeTags   bool
 | 
				
			||||||
	TagNames      []string
 | 
						TagNames      []string
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -246,6 +249,9 @@ func (opts *FindReleasesOptions) toConds(repoID int64) builder.Cond {
 | 
				
			||||||
	if !opts.IncludeDrafts {
 | 
						if !opts.IncludeDrafts {
 | 
				
			||||||
		cond = cond.And(builder.Eq{"is_draft": false})
 | 
							cond = cond.And(builder.Eq{"is_draft": false})
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if !opts.IncludeTags {
 | 
				
			||||||
 | 
							cond = cond.And(builder.Eq{"is_tag": false})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	if len(opts.TagNames) > 0 {
 | 
						if len(opts.TagNames) > 0 {
 | 
				
			||||||
		cond = cond.And(builder.In("tag_name", opts.TagNames))
 | 
							cond = cond.And(builder.In("tag_name", opts.TagNames))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -361,6 +367,8 @@ func UpdateRelease(gitRepo *git.Repository, rel *Release, attachmentUUIDs []stri
 | 
				
			||||||
	if err = createTag(gitRepo, rel); err != nil {
 | 
						if err = createTag(gitRepo, rel); err != nil {
 | 
				
			||||||
		return err
 | 
							return err
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						rel.LowerTagName = strings.ToLower(rel.TagName)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	_, err = x.Id(rel.ID).AllCols().Update(rel)
 | 
						_, err = x.Id(rel.ID).AllCols().Update(rel)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		return err
 | 
							return err
 | 
				
			||||||
| 
						 | 
					@ -397,11 +405,64 @@ func DeleteReleaseByID(id int64, u *User, delTag bool) error {
 | 
				
			||||||
		if err != nil && !strings.Contains(stderr, "not found") {
 | 
							if err != nil && !strings.Contains(stderr, "not found") {
 | 
				
			||||||
			return fmt.Errorf("git tag -d: %v - %s", err, stderr)
 | 
								return fmt.Errorf("git tag -d: %v - %s", err, stderr)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if _, err = x.Id(rel.ID).Delete(new(Release)); err != nil {
 | 
							if _, err = x.Id(rel.ID).Delete(new(Release)); err != nil {
 | 
				
			||||||
		return fmt.Errorf("Delete: %v", err)
 | 
								return fmt.Errorf("Delete: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							rel.IsTag = true
 | 
				
			||||||
 | 
							rel.IsDraft = false
 | 
				
			||||||
 | 
							rel.IsPrerelease = false
 | 
				
			||||||
 | 
							rel.Title = ""
 | 
				
			||||||
 | 
							rel.Note = ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if _, err = x.Id(rel.ID).AllCols().Update(rel); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("Update: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return nil
 | 
						return nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SyncReleasesWithTags synchronizes release table with repository tags
 | 
				
			||||||
 | 
					func SyncReleasesWithTags(repo *Repository, gitRepo *git.Repository) error {
 | 
				
			||||||
 | 
						existingRelTags := make(map[string]struct{})
 | 
				
			||||||
 | 
						opts := FindReleasesOptions{IncludeDrafts: true, IncludeTags: true}
 | 
				
			||||||
 | 
						for page := 1; ; page++ {
 | 
				
			||||||
 | 
							rels, err := GetReleasesByRepoID(repo.ID, opts, page, 100)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("GetReleasesByRepoID: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if len(rels) == 0 {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							for _, rel := range rels {
 | 
				
			||||||
 | 
								if rel.IsDraft {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								commitID, err := gitRepo.GetTagCommitID(rel.TagName)
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									return fmt.Errorf("GetTagCommitID: %v", err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if !gitRepo.IsTagExist(rel.TagName) || commitID != rel.Sha1 {
 | 
				
			||||||
 | 
									if err := pushUpdateDeleteTag(repo, gitRepo, rel.TagName); err != nil {
 | 
				
			||||||
 | 
										return fmt.Errorf("pushUpdateDeleteTag: %v", err)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									existingRelTags[strings.ToLower(rel.TagName)] = struct{}{}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						tags, err := gitRepo.GetTags()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return fmt.Errorf("GetTags: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, tagName := range tags {
 | 
				
			||||||
 | 
							if _, ok := existingRelTags[strings.ToLower(tagName)]; !ok {
 | 
				
			||||||
 | 
								if err := pushUpdateAddTag(repo, gitRepo, tagName); err != nil {
 | 
				
			||||||
 | 
									return fmt.Errorf("pushUpdateAddTag: %v", err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -940,6 +940,10 @@ func MigrateRepository(doer, u *User, opts MigrateRepoOptions) (*Repository, err
 | 
				
			||||||
		if headBranch != nil {
 | 
							if headBranch != nil {
 | 
				
			||||||
			repo.DefaultBranch = headBranch.Name
 | 
								repo.DefaultBranch = headBranch.Name
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if err = SyncReleasesWithTags(repo, gitRepo); err != nil {
 | 
				
			||||||
 | 
								log.Error(4, "Failed to synchronize tags to releases for repository: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if err = repo.UpdateSize(); err != nil {
 | 
						if err = repo.UpdateSize(); err != nil {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,6 +13,7 @@ import (
 | 
				
			||||||
	"github.com/go-xorm/xorm"
 | 
						"github.com/go-xorm/xorm"
 | 
				
			||||||
	"gopkg.in/ini.v1"
 | 
						"gopkg.in/ini.v1"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/git"
 | 
				
			||||||
	"code.gitea.io/gitea/modules/log"
 | 
						"code.gitea.io/gitea/modules/log"
 | 
				
			||||||
	"code.gitea.io/gitea/modules/process"
 | 
						"code.gitea.io/gitea/modules/process"
 | 
				
			||||||
	"code.gitea.io/gitea/modules/setting"
 | 
						"code.gitea.io/gitea/modules/setting"
 | 
				
			||||||
| 
						 | 
					@ -156,6 +157,15 @@ func (m *Mirror) runSync() bool {
 | 
				
			||||||
		return false
 | 
							return false
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						gitRepo, err := git.OpenRepository(repoPath)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							log.Error(4, "OpenRepository: %v", err)
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if err = SyncReleasesWithTags(m.Repo, gitRepo); err != nil {
 | 
				
			||||||
 | 
							log.Error(4, "Failed to synchronize tags to releases for repository: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if err := m.Repo.UpdateSize(); err != nil {
 | 
						if err := m.Repo.UpdateSize(); err != nil {
 | 
				
			||||||
		log.Error(4, "Failed to update size for mirror repository: %v", err)
 | 
							log.Error(4, "Failed to update size for mirror repository: %v", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										107
									
								
								models/update.go
									
										
									
									
									
								
							
							
						
						
									
										107
									
								
								models/update.go
									
										
									
									
									
								
							| 
						 | 
					@ -81,6 +81,93 @@ func PushUpdate(branch string, opt PushUpdateOptions) error {
 | 
				
			||||||
	return nil
 | 
						return nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func pushUpdateDeleteTag(repo *Repository, gitRepo *git.Repository, tagName string) error {
 | 
				
			||||||
 | 
						rel, err := GetRelease(repo.ID, tagName)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if IsErrReleaseNotExist(err) {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return fmt.Errorf("GetRelease: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if rel.IsTag {
 | 
				
			||||||
 | 
							if _, err = x.Id(rel.ID).Delete(new(Release)); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("Delete: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							rel.IsDraft = true
 | 
				
			||||||
 | 
							rel.NumCommits = 0
 | 
				
			||||||
 | 
							rel.Sha1 = ""
 | 
				
			||||||
 | 
							if _, err = x.Id(rel.ID).AllCols().Update(rel); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("Update: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func pushUpdateAddTag(repo *Repository, gitRepo *git.Repository, tagName string) error {
 | 
				
			||||||
 | 
						rel, err := GetRelease(repo.ID, tagName)
 | 
				
			||||||
 | 
						if err != nil && !IsErrReleaseNotExist(err) {
 | 
				
			||||||
 | 
							return fmt.Errorf("GetRelease: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tag, err := gitRepo.GetTag(tagName)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return fmt.Errorf("GetTag: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						commit, err := tag.Commit()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return fmt.Errorf("Commit: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						tagCreatedUnix := commit.Author.When.Unix()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						author, err := GetUserByEmail(commit.Author.Email)
 | 
				
			||||||
 | 
						if err != nil && !IsErrUserNotExist(err) {
 | 
				
			||||||
 | 
							return fmt.Errorf("GetUserByEmail: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						commitsCount, err := commit.CommitsCount()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return fmt.Errorf("CommitsCount: %v", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if rel == nil {
 | 
				
			||||||
 | 
							rel = &Release{
 | 
				
			||||||
 | 
								RepoID:       repo.ID,
 | 
				
			||||||
 | 
								Title:        "",
 | 
				
			||||||
 | 
								TagName:      tagName,
 | 
				
			||||||
 | 
								LowerTagName: strings.ToLower(tagName),
 | 
				
			||||||
 | 
								Target:       "",
 | 
				
			||||||
 | 
								Sha1:         commit.ID.String(),
 | 
				
			||||||
 | 
								NumCommits:   commitsCount,
 | 
				
			||||||
 | 
								Note:         "",
 | 
				
			||||||
 | 
								IsDraft:      false,
 | 
				
			||||||
 | 
								IsPrerelease: false,
 | 
				
			||||||
 | 
								IsTag:        true,
 | 
				
			||||||
 | 
								CreatedUnix:  tagCreatedUnix,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if author != nil {
 | 
				
			||||||
 | 
								rel.PublisherID = author.ID
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if _, err = x.InsertOne(rel); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("InsertOne: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							rel.Sha1 = commit.ID.String()
 | 
				
			||||||
 | 
							rel.CreatedUnix = tagCreatedUnix
 | 
				
			||||||
 | 
							rel.NumCommits = commitsCount
 | 
				
			||||||
 | 
							rel.IsDraft = false
 | 
				
			||||||
 | 
							if rel.IsTag && author != nil {
 | 
				
			||||||
 | 
								rel.PublisherID = author.ID
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if _, err = x.Id(rel.ID).AllCols().Update(rel); err != nil {
 | 
				
			||||||
 | 
								return fmt.Errorf("Update: %v", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func pushUpdate(opts PushUpdateOptions) (repo *Repository, err error) {
 | 
					func pushUpdate(opts PushUpdateOptions) (repo *Repository, err error) {
 | 
				
			||||||
	isNewRef := opts.OldCommitID == git.EmptySHA
 | 
						isNewRef := opts.OldCommitID == git.EmptySHA
 | 
				
			||||||
	isDelRef := opts.NewCommitID == git.EmptySHA
 | 
						isDelRef := opts.NewCommitID == git.EmptySHA
 | 
				
			||||||
| 
						 | 
					@ -106,23 +193,31 @@ func pushUpdate(opts PushUpdateOptions) (repo *Repository, err error) {
 | 
				
			||||||
		return nil, fmt.Errorf("GetRepositoryByName: %v", err)
 | 
							return nil, fmt.Errorf("GetRepositoryByName: %v", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if isDelRef {
 | 
					 | 
				
			||||||
		log.GitLogger.Info("Reference '%s' has been deleted from '%s/%s' by %s",
 | 
					 | 
				
			||||||
			opts.RefFullName, opts.RepoUserName, opts.RepoName, opts.PusherName)
 | 
					 | 
				
			||||||
		return repo, nil
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	gitRepo, err := git.OpenRepository(repoPath)
 | 
						gitRepo, err := git.OpenRepository(repoPath)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		return nil, fmt.Errorf("OpenRepository: %v", err)
 | 
							return nil, fmt.Errorf("OpenRepository: %v", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if isDelRef {
 | 
				
			||||||
 | 
							// Tag has been deleted
 | 
				
			||||||
 | 
							if strings.HasPrefix(opts.RefFullName, git.TagPrefix) {
 | 
				
			||||||
 | 
								err = pushUpdateDeleteTag(repo, gitRepo, opts.RefFullName[len(git.TagPrefix):])
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									return nil, fmt.Errorf("pushUpdateDeleteTag: %v", err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							log.GitLogger.Info("Reference '%s' has been deleted from '%s/%s' by %s",
 | 
				
			||||||
 | 
								opts.RefFullName, opts.RepoUserName, opts.RepoName, opts.PusherName)
 | 
				
			||||||
 | 
							return repo, nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if err = repo.UpdateSize(); err != nil {
 | 
						if err = repo.UpdateSize(); err != nil {
 | 
				
			||||||
		log.Error(4, "Failed to update size for repository: %v", err)
 | 
							log.Error(4, "Failed to update size for repository: %v", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Push tags.
 | 
						// Push tags.
 | 
				
			||||||
	if strings.HasPrefix(opts.RefFullName, git.TagPrefix) {
 | 
						if strings.HasPrefix(opts.RefFullName, git.TagPrefix) {
 | 
				
			||||||
 | 
							pushUpdateAddTag(repo, gitRepo, opts.RefFullName[len(git.TagPrefix):])
 | 
				
			||||||
		if err := CommitRepoAction(CommitRepoActionOptions{
 | 
							if err := CommitRepoAction(CommitRepoActionOptions{
 | 
				
			||||||
			PusherName:  opts.PusherName,
 | 
								PusherName:  opts.PusherName,
 | 
				
			||||||
			RepoOwnerID: owner.ID,
 | 
								RepoOwnerID: owner.ID,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -357,6 +357,7 @@ func RepoAssignment() macaron.Handler {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		count, err := models.GetReleaseCountByRepoID(ctx.Repo.Repository.ID, models.FindReleasesOptions{
 | 
							count, err := models.GetReleaseCountByRepoID(ctx.Repo.Repository.ID, models.FindReleasesOptions{
 | 
				
			||||||
			IncludeDrafts: false,
 | 
								IncludeDrafts: false,
 | 
				
			||||||
 | 
								IncludeTags:   true,
 | 
				
			||||||
		})
 | 
							})
 | 
				
			||||||
		if err != nil {
 | 
							if err != nil {
 | 
				
			||||||
			ctx.Handle(500, "GetReleaseCountByRepoID", err)
 | 
								ctx.Handle(500, "GetReleaseCountByRepoID", err)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -5,8 +5,6 @@
 | 
				
			||||||
package repo
 | 
					package repo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
	"strings"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	api "code.gitea.io/sdk/gitea"
 | 
						api "code.gitea.io/sdk/gitea"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"code.gitea.io/gitea/models"
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
| 
						 | 
					@ -36,6 +34,7 @@ func GetRelease(ctx *context.APIContext) {
 | 
				
			||||||
func ListReleases(ctx *context.APIContext) {
 | 
					func ListReleases(ctx *context.APIContext) {
 | 
				
			||||||
	releases, err := models.GetReleasesByRepoID(ctx.Repo.Repository.ID, models.FindReleasesOptions{
 | 
						releases, err := models.GetReleasesByRepoID(ctx.Repo.Repository.ID, models.FindReleasesOptions{
 | 
				
			||||||
		IncludeDrafts: ctx.Repo.AccessMode >= models.AccessModeWrite,
 | 
							IncludeDrafts: ctx.Repo.AccessMode >= models.AccessModeWrite,
 | 
				
			||||||
 | 
							IncludeTags:   false,
 | 
				
			||||||
	}, 1, 2147483647)
 | 
						}, 1, 2147483647)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		ctx.Error(500, "GetReleasesByRepoID", err)
 | 
							ctx.Error(500, "GetReleasesByRepoID", err)
 | 
				
			||||||
| 
						 | 
					@ -62,43 +61,49 @@ func CreateRelease(ctx *context.APIContext, form api.CreateReleaseOption) {
 | 
				
			||||||
		ctx.Status(404)
 | 
							ctx.Status(404)
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	tag, err := ctx.Repo.GitRepo.GetTag(form.TagName)
 | 
						rel, err := models.GetRelease(ctx.Repo.Repository.ID, form.TagName)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		ctx.Error(500, "GetTag", err)
 | 
							if !models.IsErrReleaseNotExist(err) {
 | 
				
			||||||
		return
 | 
								ctx.Handle(500, "GetRelease", err)
 | 
				
			||||||
	}
 | 
								return
 | 
				
			||||||
	commit, err := tag.Commit()
 | 
							}
 | 
				
			||||||
	if err != nil {
 | 
							rel = &models.Release{
 | 
				
			||||||
		ctx.Error(500, "Commit", err)
 | 
								RepoID:       ctx.Repo.Repository.ID,
 | 
				
			||||||
		return
 | 
								PublisherID:  ctx.User.ID,
 | 
				
			||||||
	}
 | 
								Publisher:    ctx.User,
 | 
				
			||||||
	commitsCount, err := commit.CommitsCount()
 | 
								TagName:      form.TagName,
 | 
				
			||||||
	if err != nil {
 | 
								Target:       form.Target,
 | 
				
			||||||
		ctx.Error(500, "CommitsCount", err)
 | 
								Title:        form.Title,
 | 
				
			||||||
		return
 | 
								Note:         form.Note,
 | 
				
			||||||
	}
 | 
								IsDraft:      form.IsDraft,
 | 
				
			||||||
	rel := &models.Release{
 | 
								IsPrerelease: form.IsPrerelease,
 | 
				
			||||||
		RepoID:       ctx.Repo.Repository.ID,
 | 
								IsTag:        false,
 | 
				
			||||||
		PublisherID:  ctx.User.ID,
 | 
							}
 | 
				
			||||||
		Publisher:    ctx.User,
 | 
							if err := models.CreateRelease(ctx.Repo.GitRepo, rel, nil); err != nil {
 | 
				
			||||||
		TagName:      form.TagName,
 | 
								if models.IsErrReleaseAlreadyExist(err) {
 | 
				
			||||||
		LowerTagName: strings.ToLower(form.TagName),
 | 
									ctx.Status(409)
 | 
				
			||||||
		Target:       form.Target,
 | 
								} else {
 | 
				
			||||||
		Title:        form.Title,
 | 
									ctx.Error(500, "CreateRelease", err)
 | 
				
			||||||
		Sha1:         commit.ID.String(),
 | 
								}
 | 
				
			||||||
		NumCommits:   commitsCount,
 | 
								return
 | 
				
			||||||
		Note:         form.Note,
 | 
							}
 | 
				
			||||||
		IsDraft:      form.IsDraft,
 | 
						} else {
 | 
				
			||||||
		IsPrerelease: form.IsPrerelease,
 | 
							if !rel.IsTag {
 | 
				
			||||||
		CreatedUnix:  commit.Author.When.Unix(),
 | 
								ctx.Status(409)
 | 
				
			||||||
	}
 | 
								return
 | 
				
			||||||
	if err := models.CreateRelease(ctx.Repo.GitRepo, rel, nil); err != nil {
 | 
							}
 | 
				
			||||||
		if models.IsErrReleaseAlreadyExist(err) {
 | 
					
 | 
				
			||||||
			ctx.Status(409)
 | 
							rel.Title = form.Title
 | 
				
			||||||
		} else {
 | 
							rel.Note = form.Note
 | 
				
			||||||
			ctx.Error(500, "CreateRelease", err)
 | 
							rel.IsDraft = form.IsDraft
 | 
				
			||||||
 | 
							rel.IsPrerelease = form.IsPrerelease
 | 
				
			||||||
 | 
							rel.PublisherID = ctx.User.ID
 | 
				
			||||||
 | 
							rel.IsTag = false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if err = models.UpdateRelease(ctx.Repo.GitRepo, rel, nil); err != nil {
 | 
				
			||||||
 | 
								ctx.Handle(500, "UpdateRelease", err)
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	ctx.JSON(201, rel.APIFormat())
 | 
						ctx.JSON(201, rel.APIFormat())
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -111,11 +116,12 @@ func EditRelease(ctx *context.APIContext, form api.EditReleaseOption) {
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	id := ctx.ParamsInt64(":id")
 | 
						id := ctx.ParamsInt64(":id")
 | 
				
			||||||
	rel, err := models.GetReleaseByID(id)
 | 
						rel, err := models.GetReleaseByID(id)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil && !models.IsErrReleaseNotExist(err) {
 | 
				
			||||||
		ctx.Error(500, "GetReleaseByID", err)
 | 
							ctx.Error(500, "GetReleaseByID", err)
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if rel.RepoID != ctx.Repo.Repository.ID {
 | 
						if err != nil && models.IsErrReleaseNotExist(err) ||
 | 
				
			||||||
 | 
							rel.IsTag || rel.RepoID != ctx.Repo.Repository.ID {
 | 
				
			||||||
		ctx.Status(404)
 | 
							ctx.Status(404)
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -162,12 +168,13 @@ func DeleteRelease(ctx *context.APIContext) {
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	id := ctx.ParamsInt64(":id")
 | 
						id := ctx.ParamsInt64(":id")
 | 
				
			||||||
	release, err := models.GetReleaseByID(id)
 | 
						rel, err := models.GetReleaseByID(id)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil && !models.IsErrReleaseNotExist(err) {
 | 
				
			||||||
		ctx.Error(500, "GetReleaseByID", err)
 | 
							ctx.Error(500, "GetReleaseByID", err)
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if release.RepoID != ctx.Repo.Repository.ID {
 | 
						if err != nil && models.IsErrReleaseNotExist(err) ||
 | 
				
			||||||
 | 
							rel.IsTag || rel.RepoID != ctx.Repo.Repository.ID {
 | 
				
			||||||
		ctx.Status(404)
 | 
							ctx.Status(404)
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -67,6 +67,7 @@ func Releases(ctx *context.Context) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	opts := models.FindReleasesOptions{
 | 
						opts := models.FindReleasesOptions{
 | 
				
			||||||
		IncludeDrafts: ctx.Repo.IsWriter(),
 | 
							IncludeDrafts: ctx.Repo.IsWriter(),
 | 
				
			||||||
 | 
							IncludeTags:   true,
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	releases, err := models.GetReleasesByRepoID(ctx.Repo.Repository.ID, opts, page, limit)
 | 
						releases, err := models.GetReleasesByRepoID(ctx.Repo.Repository.ID, opts, page, limit)
 | 
				
			||||||
| 
						 | 
					@ -145,57 +146,61 @@ func NewReleasePost(ctx *context.Context, form auth.NewReleaseForm) {
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var tagCreatedUnix int64
 | 
					 | 
				
			||||||
	tag, err := ctx.Repo.GitRepo.GetTag(form.TagName)
 | 
					 | 
				
			||||||
	if err == nil {
 | 
					 | 
				
			||||||
		commit, err := tag.Commit()
 | 
					 | 
				
			||||||
		if err == nil {
 | 
					 | 
				
			||||||
			tagCreatedUnix = commit.Author.When.Unix()
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	commit, err := ctx.Repo.GitRepo.GetBranchCommit(form.Target)
 | 
					 | 
				
			||||||
	if err != nil {
 | 
					 | 
				
			||||||
		ctx.Handle(500, "GetBranchCommit", err)
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	commitsCount, err := commit.CommitsCount()
 | 
					 | 
				
			||||||
	if err != nil {
 | 
					 | 
				
			||||||
		ctx.Handle(500, "CommitsCount", err)
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	rel := &models.Release{
 | 
					 | 
				
			||||||
		RepoID:       ctx.Repo.Repository.ID,
 | 
					 | 
				
			||||||
		PublisherID:  ctx.User.ID,
 | 
					 | 
				
			||||||
		Title:        form.Title,
 | 
					 | 
				
			||||||
		TagName:      form.TagName,
 | 
					 | 
				
			||||||
		Target:       form.Target,
 | 
					 | 
				
			||||||
		Sha1:         commit.ID.String(),
 | 
					 | 
				
			||||||
		NumCommits:   commitsCount,
 | 
					 | 
				
			||||||
		Note:         form.Content,
 | 
					 | 
				
			||||||
		IsDraft:      len(form.Draft) > 0,
 | 
					 | 
				
			||||||
		IsPrerelease: form.Prerelease,
 | 
					 | 
				
			||||||
		CreatedUnix:  tagCreatedUnix,
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	var attachmentUUIDs []string
 | 
						var attachmentUUIDs []string
 | 
				
			||||||
	if setting.AttachmentEnabled {
 | 
						if setting.AttachmentEnabled {
 | 
				
			||||||
		attachmentUUIDs = form.Files
 | 
							attachmentUUIDs = form.Files
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if err = models.CreateRelease(ctx.Repo.GitRepo, rel, attachmentUUIDs); err != nil {
 | 
						rel, err := models.GetRelease(ctx.Repo.Repository.ID, form.TagName)
 | 
				
			||||||
		ctx.Data["Err_TagName"] = true
 | 
						if err != nil {
 | 
				
			||||||
		switch {
 | 
							if !models.IsErrReleaseNotExist(err) {
 | 
				
			||||||
		case models.IsErrReleaseAlreadyExist(err):
 | 
								ctx.Handle(500, "GetRelease", err)
 | 
				
			||||||
			ctx.RenderWithErr(ctx.Tr("repo.release.tag_name_already_exist"), tplReleaseNew, &form)
 | 
								return
 | 
				
			||||||
		case models.IsErrInvalidTagName(err):
 | 
							}
 | 
				
			||||||
			ctx.RenderWithErr(ctx.Tr("repo.release.tag_name_invalid"), tplReleaseNew, &form)
 | 
					
 | 
				
			||||||
		default:
 | 
							rel := &models.Release{
 | 
				
			||||||
			ctx.Handle(500, "CreateRelease", err)
 | 
								RepoID:       ctx.Repo.Repository.ID,
 | 
				
			||||||
 | 
								PublisherID:  ctx.User.ID,
 | 
				
			||||||
 | 
								Title:        form.Title,
 | 
				
			||||||
 | 
								TagName:      form.TagName,
 | 
				
			||||||
 | 
								Target:       form.Target,
 | 
				
			||||||
 | 
								Note:         form.Content,
 | 
				
			||||||
 | 
								IsDraft:      len(form.Draft) > 0,
 | 
				
			||||||
 | 
								IsPrerelease: form.Prerelease,
 | 
				
			||||||
 | 
								IsTag:        false,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if err = models.CreateRelease(ctx.Repo.GitRepo, rel, attachmentUUIDs); err != nil {
 | 
				
			||||||
 | 
								ctx.Data["Err_TagName"] = true
 | 
				
			||||||
 | 
								switch {
 | 
				
			||||||
 | 
								case models.IsErrReleaseAlreadyExist(err):
 | 
				
			||||||
 | 
									ctx.RenderWithErr(ctx.Tr("repo.release.tag_name_already_exist"), tplReleaseNew, &form)
 | 
				
			||||||
 | 
								case models.IsErrInvalidTagName(err):
 | 
				
			||||||
 | 
									ctx.RenderWithErr(ctx.Tr("repo.release.tag_name_invalid"), tplReleaseNew, &form)
 | 
				
			||||||
 | 
								default:
 | 
				
			||||||
 | 
									ctx.Handle(500, "CreateRelease", err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							if !rel.IsTag {
 | 
				
			||||||
 | 
								ctx.Data["Err_TagName"] = true
 | 
				
			||||||
 | 
								ctx.RenderWithErr(ctx.Tr("repo.release.tag_name_already_exist"), tplReleaseNew, &form)
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							rel.Title = form.Title
 | 
				
			||||||
 | 
							rel.Note = form.Content
 | 
				
			||||||
 | 
							rel.IsDraft = len(form.Draft) > 0
 | 
				
			||||||
 | 
							rel.IsPrerelease = form.Prerelease
 | 
				
			||||||
 | 
							rel.PublisherID = ctx.User.ID
 | 
				
			||||||
 | 
							rel.IsTag = false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if err = models.UpdateRelease(ctx.Repo.GitRepo, rel, attachmentUUIDs); err != nil {
 | 
				
			||||||
 | 
								ctx.Data["Err_TagName"] = true
 | 
				
			||||||
 | 
								ctx.Handle(500, "UpdateRelease", err)
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	log.Trace("Release created: %s/%s:%s", ctx.User.LowerName, ctx.Repo.Repository.Name, form.TagName)
 | 
						log.Trace("Release created: %s/%s:%s", ctx.User.LowerName, ctx.Repo.Repository.Name, form.TagName)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -246,6 +251,10 @@ func EditReleasePost(ctx *context.Context, form auth.EditReleaseForm) {
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if rel.IsTag {
 | 
				
			||||||
 | 
							ctx.Handle(404, "GetRelease", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	ctx.Data["tag_name"] = rel.TagName
 | 
						ctx.Data["tag_name"] = rel.TagName
 | 
				
			||||||
	ctx.Data["tag_target"] = rel.Target
 | 
						ctx.Data["tag_target"] = rel.Target
 | 
				
			||||||
	ctx.Data["title"] = rel.Title
 | 
						ctx.Data["title"] = rel.Title
 | 
				
			||||||
| 
						 | 
					@ -275,8 +284,7 @@ func EditReleasePost(ctx *context.Context, form auth.EditReleaseForm) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// DeleteRelease delete a release
 | 
					// DeleteRelease delete a release
 | 
				
			||||||
func DeleteRelease(ctx *context.Context) {
 | 
					func DeleteRelease(ctx *context.Context) {
 | 
				
			||||||
	delTag := ctx.QueryBool("delTag")
 | 
						if err := models.DeleteReleaseByID(ctx.QueryInt64("id"), ctx.User, true); err != nil {
 | 
				
			||||||
	if err := models.DeleteReleaseByID(ctx.QueryInt64("id"), ctx.User, delTag); err != nil {
 | 
					 | 
				
			||||||
		ctx.Flash.Error("DeleteReleaseByID: " + err.Error())
 | 
							ctx.Flash.Error("DeleteReleaseByID: " + err.Error())
 | 
				
			||||||
	} else {
 | 
						} else {
 | 
				
			||||||
		ctx.Flash.Success(ctx.Tr("repo.release.deletion_success"))
 | 
							ctx.Flash.Success(ctx.Tr("repo.release.deletion_success"))
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -17,7 +17,9 @@
 | 
				
			||||||
			{{range .Releases}}
 | 
								{{range .Releases}}
 | 
				
			||||||
				<li class="ui grid">
 | 
									<li class="ui grid">
 | 
				
			||||||
					<div class="ui four wide column meta">
 | 
										<div class="ui four wide column meta">
 | 
				
			||||||
						{{if .PublisherID}}
 | 
											{{if .IsTag}}
 | 
				
			||||||
 | 
												{{if .Created}}<span class="time">{{TimeSince .Created $.Lang}}</span>{{end}}
 | 
				
			||||||
 | 
											{{else}}
 | 
				
			||||||
							{{if .IsDraft}}
 | 
												{{if .IsDraft}}
 | 
				
			||||||
								<span class="ui yellow label">{{$.i18n.Tr "repo.release.draft"}}</span>
 | 
													<span class="ui yellow label">{{$.i18n.Tr "repo.release.draft"}}</span>
 | 
				
			||||||
							{{else if .IsPrerelease}}
 | 
												{{else if .IsPrerelease}}
 | 
				
			||||||
| 
						 | 
					@ -28,13 +30,22 @@
 | 
				
			||||||
							<span class="tag text blue">
 | 
												<span class="tag text blue">
 | 
				
			||||||
								<a href="{{$.RepoLink}}/src/{{.TagName}}" rel="nofollow"><i class="tag icon"></i> {{.TagName}}</a>
 | 
													<a href="{{$.RepoLink}}/src/{{.TagName}}" rel="nofollow"><i class="tag icon"></i> {{.TagName}}</a>
 | 
				
			||||||
							</span>
 | 
												</span>
 | 
				
			||||||
 | 
												<span class="commit">
 | 
				
			||||||
 | 
													<a href="{{$.RepoLink}}/src/{{.Sha1}}" rel="nofollow"><i class="code icon"></i> {{ShortSha .Sha1}}</a>
 | 
				
			||||||
 | 
												</span>
 | 
				
			||||||
						{{end}}
 | 
											{{end}}
 | 
				
			||||||
						<span class="commit">
 | 
					 | 
				
			||||||
							<a href="{{$.RepoLink}}/src/{{.Sha1}}" rel="nofollow"><i class="code icon"></i> {{ShortSha .Sha1}}</a>
 | 
					 | 
				
			||||||
						</span>
 | 
					 | 
				
			||||||
					</div>
 | 
										</div>
 | 
				
			||||||
					<div class="ui twelve wide column detail">
 | 
										<div class="ui twelve wide column detail">
 | 
				
			||||||
						{{if .PublisherID}}
 | 
											{{if .IsTag}}
 | 
				
			||||||
 | 
												<h4>
 | 
				
			||||||
 | 
													<a href="{{$.RepoLink}}/src/{{.TagName}}" rel="nofollow"><i class="tag icon"></i> {{.TagName}}</a>
 | 
				
			||||||
 | 
												</h4>
 | 
				
			||||||
 | 
												<div class="download">
 | 
				
			||||||
 | 
													<a href="{{$.RepoLink}}/src/{{.Sha1}}" rel="nofollow"><i class="code icon"></i> {{ShortSha .Sha1}}</a>
 | 
				
			||||||
 | 
													<a href="{{$.RepoLink}}/archive/{{.TagName}}.zip" rel="nofollow"><i class="octicon octicon-file-zip"></i> ZIP</a>
 | 
				
			||||||
 | 
													<a href="{{$.RepoLink}}/archive/{{.TagName}}.tar.gz"><i class="octicon octicon-file-zip"></i> TAR.GZ</a>
 | 
				
			||||||
 | 
												</div>
 | 
				
			||||||
 | 
											{{else}}
 | 
				
			||||||
							<h3>
 | 
												<h3>
 | 
				
			||||||
								<a href="{{$.RepoLink}}/src/{{.TagName}}">{{.Title}}</a>
 | 
													<a href="{{$.RepoLink}}/src/{{.TagName}}">{{.Title}}</a>
 | 
				
			||||||
								{{if $.IsRepositoryWriter}}<small>(<a href="{{$.RepoLink}}/releases/edit/{{.TagName}}" rel="nofollow">{{$.i18n.Tr "repo.release.edit"}}</a>)</small>{{end}}
 | 
													{{if $.IsRepositoryWriter}}<small>(<a href="{{$.RepoLink}}/releases/edit/{{.TagName}}" rel="nofollow">{{$.i18n.Tr "repo.release.edit"}}</a>)</small>{{end}}
 | 
				
			||||||
| 
						 | 
					@ -70,14 +81,6 @@
 | 
				
			||||||
									{{end}}
 | 
														{{end}}
 | 
				
			||||||
								</ul>
 | 
													</ul>
 | 
				
			||||||
							</div>
 | 
												</div>
 | 
				
			||||||
						{{else}}
 | 
					 | 
				
			||||||
							<h4>
 | 
					 | 
				
			||||||
								<a href="{{$.RepoLink}}/src/{{.TagName}}" rel="nofollow"><i class="tag icon"></i> {{.TagName}}</a>
 | 
					 | 
				
			||||||
							</h4>
 | 
					 | 
				
			||||||
							<div class="download">
 | 
					 | 
				
			||||||
								<a href="{{$.RepoLink}}/archive/{{.TagName}}.zip" rel="nofollow"><i class="octicon octicon-file-zip"></i> ZIP</a>
 | 
					 | 
				
			||||||
								<a href="{{$.RepoLink}}/archive/{{.TagName}}.tar.gz"><i class="octicon octicon-file-zip"></i> TAR.GZ</a>
 | 
					 | 
				
			||||||
							</div>
 | 
					 | 
				
			||||||
						{{end}}
 | 
											{{end}}
 | 
				
			||||||
						<span class="dot"> </span>
 | 
											<span class="dot"> </span>
 | 
				
			||||||
					</div>
 | 
										</div>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue