0
Fork 0
mirror of https://codeberg.org/forgejo/forgejo.git synced 2025-01-15 02:44:56 -05:00
forgejo/models/release.go

223 lines
5.5 KiB
Go
Raw Normal View History

2014-04-14 00:57:25 -05:00
// Copyright 2014 The Gogs 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 models
import (
2015-11-20 02:38:41 -05:00
"fmt"
2014-06-12 16:47:23 -05:00
"sort"
2014-04-14 00:57:25 -05:00
"strings"
"time"
2015-08-24 08:01:23 -05:00
"github.com/go-xorm/xorm"
"code.gitea.io/git"
"code.gitea.io/gitea/modules/process"
2014-04-14 00:57:25 -05:00
)
// Release represents a release of repository.
type Release struct {
2015-11-15 23:52:46 -05:00
ID int64 `xorm:"pk autoincr"`
RepoID int64
PublisherID int64
2014-04-14 00:57:25 -05:00
Publisher *User `xorm:"-"`
TagName string
LowerTagName string
2014-06-12 08:10:39 -05:00
Target string
2014-06-12 16:47:23 -05:00
Title string
2014-06-12 08:10:39 -05:00
Sha1 string `xorm:"VARCHAR(40)"`
NumCommits int64
NumCommitsBehind int64 `xorm:"-"`
2014-04-14 00:57:25 -05:00
Note string `xorm:"TEXT"`
2014-06-12 16:47:23 -05:00
IsDraft bool `xorm:"NOT NULL DEFAULT false"`
2014-04-14 00:57:25 -05:00
IsPrerelease bool
Created time.Time `xorm:"-"`
CreatedUnix int64
}
// BeforeInsert is invoked from XORM before inserting an object of this type.
func (r *Release) BeforeInsert() {
if r.CreatedUnix == 0 {
r.CreatedUnix = time.Now().Unix()
}
2014-04-14 00:57:25 -05:00
}
// AfterSet is invoked from XORM after setting the value of a field of this object.
2015-08-24 08:01:23 -05:00
func (r *Release) AfterSet(colName string, _ xorm.Cell) {
switch colName {
case "created_unix":
r.Created = time.Unix(r.CreatedUnix, 0).Local()
2015-08-24 08:01:23 -05:00
}
}
2014-04-14 00:57:25 -05:00
// IsReleaseExist returns true if release with given tag name already exists.
2015-11-15 23:52:46 -05:00
func IsReleaseExist(repoID int64, tagName string) (bool, error) {
2014-04-14 00:57:25 -05:00
if len(tagName) == 0 {
return false, nil
}
2015-11-15 23:52:46 -05:00
return x.Get(&Release{RepoID: repoID, LowerTagName: strings.ToLower(tagName)})
2014-04-14 00:57:25 -05:00
}
2014-06-12 16:47:23 -05:00
func createTag(gitRepo *git.Repository, rel *Release) error {
// Only actual create when publish.
if !rel.IsDraft {
if !gitRepo.IsTagExist(rel.TagName) {
commit, err := gitRepo.GetBranchCommit(rel.Target)
2014-06-12 16:47:23 -05:00
if err != nil {
return fmt.Errorf("GetBranchCommit: %v", err)
2014-06-12 16:47:23 -05:00
}
// Trim '--' prefix to prevent command line argument vulnerability.
rel.TagName = strings.TrimPrefix(rel.TagName, "--")
2015-11-03 22:49:06 -05:00
if err = gitRepo.CreateTag(rel.TagName, commit.ID.String()); err != nil {
if strings.Contains(err.Error(), "is not a valid tag name") {
return ErrInvalidTagName{rel.TagName}
}
2014-06-12 16:47:23 -05:00
return err
}
} else {
commit, err := gitRepo.GetTagCommit(rel.TagName)
2014-06-12 16:47:23 -05:00
if err != nil {
return fmt.Errorf("GetTagCommit: %v", err)
2014-06-12 16:47:23 -05:00
}
rel.Sha1 = commit.ID.String()
2014-06-12 16:47:23 -05:00
rel.NumCommits, err = commit.CommitsCount()
if err != nil {
return fmt.Errorf("CommitsCount: %v", err)
2014-06-12 16:47:23 -05:00
}
}
}
return nil
}
2014-04-14 00:57:25 -05:00
// CreateRelease creates a new release of repository.
2014-05-09 22:14:24 -05:00
func CreateRelease(gitRepo *git.Repository, rel *Release) error {
2015-11-15 23:52:46 -05:00
isExist, err := IsReleaseExist(rel.RepoID, rel.TagName)
2014-04-14 00:57:25 -05:00
if err != nil {
return err
} else if isExist {
2015-11-15 23:52:46 -05:00
return ErrReleaseAlreadyExist{rel.TagName}
2014-04-14 00:57:25 -05:00
}
2014-06-12 16:47:23 -05:00
if err = createTag(gitRepo, rel); err != nil {
return err
}
rel.LowerTagName = strings.ToLower(rel.TagName)
2014-06-20 23:51:41 -05:00
_, err = x.InsertOne(rel)
2014-06-12 16:47:23 -05:00
return err
}
2014-06-12 08:10:39 -05:00
2014-06-12 16:47:23 -05:00
// GetRelease returns release by given ID.
2015-11-15 23:52:46 -05:00
func GetRelease(repoID int64, tagName string) (*Release, error) {
isExist, err := IsReleaseExist(repoID, tagName)
2014-06-12 16:47:23 -05:00
if err != nil {
return nil, err
} else if !isExist {
2015-11-20 02:38:41 -05:00
return nil, ErrReleaseNotExist{0, tagName}
2014-06-12 16:47:23 -05:00
}
2014-04-14 00:57:25 -05:00
2015-11-15 23:52:46 -05:00
rel := &Release{RepoID: repoID, LowerTagName: strings.ToLower(tagName)}
2014-06-20 23:51:41 -05:00
_, err = x.Get(rel)
2014-06-12 16:47:23 -05:00
return rel, err
}
2015-11-20 02:38:41 -05:00
// GetReleaseByID returns release with given ID.
func GetReleaseByID(id int64) (*Release, error) {
rel := new(Release)
2016-11-10 10:16:32 -05:00
has, err := x.
Id(id).
Get(rel)
2015-11-20 02:38:41 -05:00
if err != nil {
return nil, err
} else if !has {
return nil, ErrReleaseNotExist{id, ""}
}
return rel, nil
}
// GetReleasesByRepoID returns a list of releases of repository.
2016-11-04 14:28:07 -05:00
func GetReleasesByRepoID(repoID int64, page, pageSize int) (rels []*Release, err error) {
if page <= 0 {
page = 1
}
2016-11-10 10:16:32 -05:00
err = x.
Desc("created_unix").
Limit(pageSize, (page-1)*pageSize).
Find(&rels, Release{RepoID: repoID})
2014-06-12 16:47:23 -05:00
return rels, err
}
type releaseSorter struct {
2014-06-12 16:47:23 -05:00
rels []*Release
}
func (rs *releaseSorter) Len() int {
2014-06-12 16:47:23 -05:00
return len(rs.rels)
}
func (rs *releaseSorter) Less(i, j int) bool {
2014-06-12 16:47:23 -05:00
diffNum := rs.rels[i].NumCommits - rs.rels[j].NumCommits
if diffNum != 0 {
return diffNum > 0
2014-04-14 00:57:25 -05:00
}
2014-06-12 16:47:23 -05:00
return rs.rels[i].Created.After(rs.rels[j].Created)
}
2014-04-14 00:57:25 -05:00
func (rs *releaseSorter) Swap(i, j int) {
2014-06-12 16:47:23 -05:00
rs.rels[i], rs.rels[j] = rs.rels[j], rs.rels[i]
}
// SortReleases sorts releases by number of commits and created time.
func SortReleases(rels []*Release) {
sorter := &releaseSorter{rels: rels}
2014-06-12 16:47:23 -05:00
sort.Sort(sorter)
}
// UpdateRelease updates information of a release.
func UpdateRelease(gitRepo *git.Repository, rel *Release) (err error) {
if err = createTag(gitRepo, rel); err != nil {
return err
}
2015-11-15 23:52:46 -05:00
_, err = x.Id(rel.ID).AllCols().Update(rel)
2014-04-14 00:57:25 -05:00
return err
}
2015-11-20 02:38:41 -05:00
// DeleteReleaseByID deletes a release and corresponding Git tag by given ID.
func DeleteReleaseByID(id int64, u *User) error {
2015-11-20 02:38:41 -05:00
rel, err := GetReleaseByID(id)
if err != nil {
return fmt.Errorf("GetReleaseByID: %v", err)
}
repo, err := GetRepositoryByID(rel.RepoID)
if err != nil {
return fmt.Errorf("GetRepositoryByID: %v", err)
}
has, err := HasAccess(u, repo, AccessModeWrite)
if err != nil {
return fmt.Errorf("HasAccess: %v", err)
} else if !has {
return fmt.Errorf("DeleteReleaseByID: permission denied")
}
2015-11-26 17:33:45 -05:00
_, stderr, err := process.ExecDir(-1, repo.RepoPath(),
fmt.Sprintf("DeleteReleaseByID (git tag -d): %d", rel.ID),
2015-11-20 02:38:41 -05:00
"git", "tag", "-d", rel.TagName)
if err != nil && !strings.Contains(stderr, "not found") {
return fmt.Errorf("git tag -d: %v - %s", err, stderr)
}
if _, err = x.Id(rel.ID).Delete(new(Release)); err != nil {
return fmt.Errorf("Delete: %v", err)
}
return nil
}