Files
Gogs/internal/database/release.go

371 lines
9.7 KiB
Go
Raw Normal View History

package database
2014-04-14 01:57:25 -04:00
import (
2015-11-20 02:38:41 -05:00
"fmt"
2014-06-12 17:47:23 -04:00
"sort"
2014-04-14 01:57:25 -04:00
"strings"
"time"
"github.com/cockroachdb/errors"
2018-05-27 08:53:48 +08:00
"github.com/gogs/git-module"
api "github.com/gogs/go-gogs-client"
"gorm.io/gorm"
log "unknwon.dev/clog/v2"
"gogs.io/gogs/internal/errutil"
"gogs.io/gogs/internal/process"
2014-04-14 01:57:25 -04:00
)
// Release represents a release of repository.
type Release struct {
ID int64
2015-11-15 23:52:46 -05:00
RepoID int64
Repo *Repository `gorm:"-" json:"-"`
2015-11-15 23:52:46 -05:00
PublisherID int64
Publisher *User `gorm:"-" json:"-"`
2014-04-14 01:57:25 -04:00
TagName string
LowerTagName string
2014-06-12 09:10:39 -04:00
Target string
2014-06-12 17:47:23 -04:00
Title string
Sha1 string `gorm:"type:varchar(40)"`
NumCommits int64
NumCommitsBehind int64 `gorm:"-" json:"-"`
Note string `gorm:"type:text"`
IsDraft bool `gorm:"not null;default:false"`
2014-04-14 01:57:25 -04:00
IsPrerelease bool
Created time.Time `gorm:"-" json:"-"`
CreatedUnix int64
Attachments []*Attachment `gorm:"-" json:"-"`
}
func (r *Release) BeforeCreate(tx *gorm.DB) error {
if r.CreatedUnix == 0 {
r.CreatedUnix = tx.NowFunc().Unix()
}
return nil
2014-04-14 01:57:25 -04:00
}
func (r *Release) AfterFind(tx *gorm.DB) error {
r.Created = time.Unix(r.CreatedUnix, 0).Local()
return nil
2015-08-24 21:01:23 +08:00
}
func (r *Release) loadAttributes(e *gorm.DB) (err error) {
2017-03-11 18:41:32 -05:00
if r.Repo == nil {
r.Repo, err = getRepositoryByID(e, r.RepoID)
if err != nil {
return errors.Newf("getRepositoryByID [repo_id: %d]: %v", r.RepoID, err)
2017-03-11 18:41:32 -05:00
}
}
if r.Publisher == nil {
r.Publisher, err = getUserByID(e, r.PublisherID)
if err != nil {
if IsErrUserNotExist(err) {
r.PublisherID = -1
r.Publisher = NewGhostUser()
} else {
return errors.Newf("getUserByID.(Publisher) [publisher_id: %d]: %v", r.PublisherID, err)
}
}
}
if r.Attachments == nil {
r.Attachments, err = getAttachmentsByReleaseID(e, r.ID)
if err != nil {
return errors.Newf("getAttachmentsByReleaseID [%d]: %v", r.ID, err)
}
}
return nil
}
func (r *Release) LoadAttributes() error {
return r.loadAttributes(db)
}
2017-03-11 18:41:32 -05:00
// This method assumes some fields assigned with values:
// Required - Publisher
func (r *Release) APIFormat() *api.Release {
return &api.Release{
ID: r.ID,
TagName: r.TagName,
TargetCommitish: r.Target,
Name: r.Title,
Body: r.Note,
Draft: r.IsDraft,
Prerelease: r.IsPrerelease,
Author: r.Publisher.APIFormat(),
Created: r.Created,
}
}
2014-04-14 01:57:25 -04: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) {
if tagName == "" {
2014-04-14 01:57:25 -04:00
return false, nil
}
var count int64
err := db.Model(&Release{}).Where("repo_id = ? AND lower_tag_name = ?", repoID, strings.ToLower(tagName)).Count(&count).Error
return count > 0, err
2014-04-14 01:57:25 -04:00
}
func createTag(gitRepo *git.Repository, r *Release) error {
2014-06-12 17:47:23 -04:00
// Only actual create when publish.
if !r.IsDraft {
git: migrate to github.com/gogs/git-module@v1.0.0 (#5958) * WIP * Finish `internal/db/git_diff.go` * FInish internal/db/mirror.go * Finish internal/db/pull.go * Finish internal/db/release.go * Finish internal/db/repo.go * Finish internal/db/repo_branch.go * Finish internal/db/repo_editor.go * Finish internal/db/update.go * Save my work * Add license header * Compile! * Merge master * Finish internal/cmd/hook.go * Finish internal/conf/static.go * Finish internal/context/repo.go * Finish internal/db/action.go * Finish internal/db/git_diff.go * Fix submodule URL inferring * Finish internal/db/mirror.go * Updat to beta.4 * css: update fonts * Finish internal/db/pull.go * Finish internal/db/release.go * Finish internal/db/repo_branch.go * Finish internal/db/wiki.go * gitutil: enhance infer submodule UR * Finish internal/route/api/v1/repo/commits.go * mirror: only collect branch commits after sync * mirror: fix tag support * Finish internal/db/repo.go * Finish internal/db/repo_editor.go * Finish internal/db/update.go * Finish internal/gitutil/pull_request.go * Make it compile * Finish internal/route/repo/setting.go * Finish internal/route/repo/branch.go * Finish internal/route/api/v1/repo/file.go * Finish internal/route/repo/download.go * Finish internal/route/repo/editor.go * Use helper * Finish internal/route/repo/issue.go * Finish internal/route/repo/pull.go * Finish internal/route/repo/release.go * Finish internal/route/repo/repo.go * Finish internal/route/repo/wiki.go * Finish internal/route/repo/commit.go * Finish internal/route/repo/view.go * Finish internal/gitutil/tag.go * go.sum
2020-03-08 19:09:31 +08:00
if !gitRepo.HasTag(r.TagName) {
commit, err := gitRepo.BranchCommit(r.Target)
2014-06-12 17:47:23 -04:00
if err != nil {
return errors.Newf("get branch commit: %v", err)
2014-06-12 17:47:23 -04:00
}
// 🚨 SECURITY: Trim any leading '-' to prevent command line argument injection.
r.TagName = strings.TrimLeft(r.TagName, "-")
if err = gitRepo.CreateTag(r.TagName, commit.ID.String()); err != nil {
if strings.Contains(err.Error(), "is not a valid tag name") {
return ErrInvalidTagName{r.TagName}
}
2014-06-12 17:47:23 -04:00
return err
}
} else {
git: migrate to github.com/gogs/git-module@v1.0.0 (#5958) * WIP * Finish `internal/db/git_diff.go` * FInish internal/db/mirror.go * Finish internal/db/pull.go * Finish internal/db/release.go * Finish internal/db/repo.go * Finish internal/db/repo_branch.go * Finish internal/db/repo_editor.go * Finish internal/db/update.go * Save my work * Add license header * Compile! * Merge master * Finish internal/cmd/hook.go * Finish internal/conf/static.go * Finish internal/context/repo.go * Finish internal/db/action.go * Finish internal/db/git_diff.go * Fix submodule URL inferring * Finish internal/db/mirror.go * Updat to beta.4 * css: update fonts * Finish internal/db/pull.go * Finish internal/db/release.go * Finish internal/db/repo_branch.go * Finish internal/db/wiki.go * gitutil: enhance infer submodule UR * Finish internal/route/api/v1/repo/commits.go * mirror: only collect branch commits after sync * mirror: fix tag support * Finish internal/db/repo.go * Finish internal/db/repo_editor.go * Finish internal/db/update.go * Finish internal/gitutil/pull_request.go * Make it compile * Finish internal/route/repo/setting.go * Finish internal/route/repo/branch.go * Finish internal/route/api/v1/repo/file.go * Finish internal/route/repo/download.go * Finish internal/route/repo/editor.go * Use helper * Finish internal/route/repo/issue.go * Finish internal/route/repo/pull.go * Finish internal/route/repo/release.go * Finish internal/route/repo/repo.go * Finish internal/route/repo/wiki.go * Finish internal/route/repo/commit.go * Finish internal/route/repo/view.go * Finish internal/gitutil/tag.go * go.sum
2020-03-08 19:09:31 +08:00
commit, err := gitRepo.TagCommit(r.TagName)
2014-06-12 17:47:23 -04:00
if err != nil {
return errors.Newf("get tag commit: %v", err)
2014-06-12 17:47:23 -04:00
}
r.Sha1 = commit.ID.String()
r.NumCommits, err = commit.CommitsCount()
2014-06-12 17:47:23 -04:00
if err != nil {
return errors.Newf("count commits: %v", err)
2014-06-12 17:47:23 -04:00
}
}
}
return nil
}
2017-03-11 18:41:32 -05:00
func (r *Release) preparePublishWebhooks() {
if err := PrepareWebhooks(r.Repo, HookEventTypeRelease, &api.ReleasePayload{
2017-03-11 18:41:32 -05:00
Action: api.HOOK_RELEASE_PUBLISHED,
Release: r.APIFormat(),
Repository: r.Repo.APIFormatLegacy(nil),
2017-03-11 18:41:32 -05:00
Sender: r.Publisher.APIFormat(),
}); err != nil {
log.Error("PrepareWebhooks: %v", err)
2017-03-11 18:41:32 -05:00
}
}
// NewRelease creates a new release with attachments for repository.
func NewRelease(gitRepo *git.Repository, r *Release, uuids []string) error {
isExist, err := IsReleaseExist(r.RepoID, r.TagName)
2014-04-14 01:57:25 -04:00
if err != nil {
return err
} else if isExist {
return ErrReleaseAlreadyExist{r.TagName}
2014-04-14 01:57:25 -04:00
}
if err = createTag(gitRepo, r); err != nil {
2014-06-12 17:47:23 -04:00
return err
}
r.LowerTagName = strings.ToLower(r.TagName)
err = db.Transaction(func(tx *gorm.DB) error {
if err := tx.Create(r).Error; err != nil {
return errors.Newf("insert: %v", err)
}
2017-03-11 18:41:32 -05:00
if len(uuids) > 0 {
if err := tx.Model(&Attachment{}).Where("uuid IN ?", uuids).Update("release_id", r.ID).Error; err != nil {
return errors.Newf("link attachments: %v", err)
}
}
return nil
})
if err != nil {
return err
}
2017-03-11 18:41:32 -05:00
// Only send webhook when actually published, skip drafts
if r.IsDraft {
return nil
}
r, err = GetReleaseByID(r.ID)
if err != nil {
return errors.Newf("GetReleaseByID: %v", err)
2017-03-11 18:41:32 -05:00
}
r.preparePublishWebhooks()
return nil
2014-06-12 17:47:23 -04:00
}
2014-06-12 09:10:39 -04:00
var _ errutil.NotFound = (*ErrReleaseNotExist)(nil)
type ErrReleaseNotExist struct {
args map[string]any
}
func IsErrReleaseNotExist(err error) bool {
_, ok := err.(ErrReleaseNotExist)
return ok
}
func (err ErrReleaseNotExist) Error() string {
return fmt.Sprintf("release does not exist: %v", err.args)
}
func (ErrReleaseNotExist) NotFound() bool {
return true
}
2014-06-12 17:47:23 -04: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 17:47:23 -04:00
if err != nil {
return nil, err
} else if !isExist {
return nil, ErrReleaseNotExist{args: map[string]any{"tag": tagName}}
2014-06-12 17:47:23 -04:00
}
2014-04-14 01:57:25 -04:00
r := &Release{}
if err = db.Where("repo_id = ? AND lower_tag_name = ?", repoID, strings.ToLower(tagName)).First(r).Error; err != nil {
return nil, errors.Newf("get: %v", err)
}
return r, r.LoadAttributes()
2014-06-12 17:47:23 -04:00
}
2015-11-20 02:38:41 -05:00
// GetReleaseByID returns release with given ID.
func GetReleaseByID(id int64) (*Release, error) {
r := new(Release)
err := db.Where("id = ?", id).First(r).Error
2015-11-20 02:38:41 -05:00
if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, ErrReleaseNotExist{args: map[string]any{"releaseID": id}}
}
2015-11-20 02:38:41 -05:00
return nil, err
}
return r, r.LoadAttributes()
}
// GetPublishedReleasesByRepoID returns a list of published releases of repository.
// If matches is not empty, only published releases in matches will be returned.
// In any case, drafts won't be returned by this function.
func GetPublishedReleasesByRepoID(repoID int64, matches ...string) ([]*Release, error) {
query := db.Where("repo_id = ? AND is_draft = ?", repoID, false).Order("created_unix DESC")
if len(matches) > 0 {
query = query.Where("tag_name IN ?", matches)
}
releases := make([]*Release, 0, 5)
return releases, query.Find(&releases).Error
2015-11-20 02:38:41 -05:00
}
// GetReleasesByRepoID returns a list of all releases (including drafts) of given repository.
func GetReleasesByRepoID(repoID int64) ([]*Release, error) {
releases := make([]*Release, 0)
return releases, db.Where("repo_id = ?", repoID).Find(&releases).Error
}
// GetDraftReleasesByRepoID returns all draft releases of repository.
func GetDraftReleasesByRepoID(repoID int64) ([]*Release, error) {
releases := make([]*Release, 0)
return releases, db.Where("repo_id = ? AND is_draft = ?", repoID, true).Find(&releases).Error
2014-06-12 17:47:23 -04:00
}
type ReleaseSorter struct {
releases []*Release
2014-06-12 17:47:23 -04:00
}
func (rs *ReleaseSorter) Len() int {
return len(rs.releases)
2014-06-12 17:47:23 -04:00
}
func (rs *ReleaseSorter) Less(i, j int) bool {
diffNum := rs.releases[i].NumCommits - rs.releases[j].NumCommits
2014-06-12 17:47:23 -04:00
if diffNum != 0 {
return diffNum > 0
2014-04-14 01:57:25 -04:00
}
return rs.releases[i].Created.After(rs.releases[j].Created)
2014-06-12 17:47:23 -04:00
}
2014-04-14 01:57:25 -04:00
2014-06-12 17:47:23 -04:00
func (rs *ReleaseSorter) Swap(i, j int) {
rs.releases[i], rs.releases[j] = rs.releases[j], rs.releases[i]
2014-06-12 17:47:23 -04:00
}
// SortReleases sorts releases by number of commits and created time.
func SortReleases(rels []*Release) {
sorter := &ReleaseSorter{releases: rels}
2014-06-12 17:47:23 -04:00
sort.Sort(sorter)
}
// UpdateRelease updates information of a release.
func UpdateRelease(doer *User, gitRepo *git.Repository, r *Release, isPublish bool, uuids []string) (err error) {
2017-03-11 18:41:32 -05:00
if err = createTag(gitRepo, r); err != nil {
return errors.Newf("createTag: %v", err)
2017-03-11 18:41:32 -05:00
}
r.PublisherID = doer.ID
err = db.Transaction(func(tx *gorm.DB) error {
if err := tx.Model(r).Where("id = ?", r.ID).Updates(r).Error; err != nil {
return errors.Newf("Update: %v", err)
}
// Unlink all current attachments and link back later if still valid
if err := tx.Exec("UPDATE attachment SET release_id = 0 WHERE release_id = ?", r.ID).Error; err != nil {
return errors.Newf("unlink current attachments: %v", err)
}
if len(uuids) > 0 {
if err := tx.Model(&Attachment{}).Where("uuid IN ?", uuids).Update("release_id", r.ID).Error; err != nil {
return errors.Newf("link attachments: %v", err)
}
}
return nil
})
if err != nil {
return err
}
2017-03-11 18:41:32 -05:00
if !isPublish {
return nil
}
r.Publisher = doer
r.preparePublishWebhooks()
return nil
2014-04-14 01:57:25 -04:00
}
2015-11-20 02:38:41 -05:00
2016-12-22 19:58:30 -05:00
// DeleteReleaseOfRepoByID deletes a release and corresponding Git tag by given ID.
func DeleteReleaseOfRepoByID(repoID, id int64) error {
2015-11-20 02:38:41 -05:00
rel, err := GetReleaseByID(id)
if err != nil {
return errors.Newf("GetReleaseByID: %v", err)
2015-11-20 02:38:41 -05:00
}
// Mark sure the delete operation against same repository.
2016-12-22 19:35:06 -05:00
if repoID != rel.RepoID {
return nil
}
2015-11-20 02:38:41 -05:00
repo, err := GetRepositoryByID(rel.RepoID)
if err != nil {
return errors.Newf("GetRepositoryByID: %v", err)
2015-11-20 02:38:41 -05:00
}
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 errors.Newf("git tag -d: %v - %s", err, stderr)
2015-11-20 02:38:41 -05:00
}
if err = db.Where("id = ?", rel.ID).Delete(new(Release)).Error; err != nil {
return errors.Newf("delete: %v", err)
2015-11-20 02:38:41 -05:00
}
return nil
}