2017-01-20 12:28:46 +05:30
// 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 models
import (
"fmt"
"path"
2021-09-19 17:19:59 +05:30
"code.gitea.io/gitea/models/db"
2017-01-20 12:28:46 +05:30
"code.gitea.io/gitea/modules/setting"
2020-08-18 09:53:45 +05:30
"code.gitea.io/gitea/modules/storage"
2019-08-15 20:16:21 +05:30
"code.gitea.io/gitea/modules/timeutil"
2018-03-06 06:52:16 +05:30
2019-10-17 14:56:49 +05:30
"xorm.io/xorm"
2017-01-20 12:28:46 +05:30
)
// Attachment represent a attachment of issue/comment/release.
type Attachment struct {
2017-04-20 08:01:31 +05:30
ID int64 ` xorm:"pk autoincr" `
UUID string ` xorm:"uuid UNIQUE" `
2021-09-08 20:49:30 +05:30
RepoID int64 ` xorm:"INDEX" ` // this should not be zero
IssueID int64 ` xorm:"INDEX" ` // maybe zero when creating
ReleaseID int64 ` xorm:"INDEX" ` // maybe zero when creating
2019-04-03 00:55:05 +05:30
UploaderID int64 ` xorm:"INDEX DEFAULT 0" ` // Notice: will be zero before this column added
2017-04-20 08:01:31 +05:30
CommentID int64
Name string
2019-08-15 20:16:21 +05:30
DownloadCount int64 ` xorm:"DEFAULT 0" `
Size int64 ` xorm:"DEFAULT 0" `
CreatedUnix timeutil . TimeStamp ` xorm:"created" `
2017-01-20 12:28:46 +05:30
}
2021-09-19 17:19:59 +05:30
func init ( ) {
db . RegisterModel ( new ( Attachment ) )
}
2017-04-20 08:01:31 +05:30
// IncreaseDownloadCount is update download count + 1
func ( a * Attachment ) IncreaseDownloadCount ( ) error {
// Update download count.
2021-09-19 17:19:59 +05:30
if _ , err := db . DefaultContext ( ) . Engine ( ) . Exec ( "UPDATE `attachment` SET download_count=download_count+1 WHERE id=?" , a . ID ) ; err != nil {
2017-04-20 08:01:31 +05:30
return fmt . Errorf ( "increase attachment count: %v" , err )
}
return nil
}
2020-08-18 09:53:45 +05:30
// AttachmentRelativePath returns the relative path
func AttachmentRelativePath ( uuid string ) string {
return path . Join ( uuid [ 0 : 1 ] , uuid [ 1 : 2 ] , uuid )
2017-01-20 12:28:46 +05:30
}
2020-08-18 09:53:45 +05:30
// RelativePath returns the relative path of the attachment
func ( a * Attachment ) RelativePath ( ) string {
return AttachmentRelativePath ( a . UUID )
2017-01-20 12:28:46 +05:30
}
2018-03-06 06:52:16 +05:30
// DownloadURL returns the download url of the attached file
func ( a * Attachment ) DownloadURL ( ) string {
return fmt . Sprintf ( "%sattachments/%s" , setting . AppURL , a . UUID )
}
2020-01-05 04:50:08 +05:30
// LinkedRepository returns the linked repo if any
func ( a * Attachment ) LinkedRepository ( ) ( * Repository , UnitType , error ) {
if a . IssueID != 0 {
iss , err := GetIssueByID ( a . IssueID )
if err != nil {
return nil , UnitTypeIssues , err
}
repo , err := GetRepositoryByID ( iss . RepoID )
2020-02-19 06:06:19 +05:30
unitType := UnitTypeIssues
if iss . IsPull {
unitType = UnitTypePullRequests
}
return repo , unitType , err
2020-01-05 04:50:08 +05:30
} else if a . ReleaseID != 0 {
rel , err := GetReleaseByID ( a . ReleaseID )
if err != nil {
return nil , UnitTypeReleases , err
}
repo , err := GetRepositoryByID ( rel . RepoID )
return repo , UnitTypeReleases , err
}
return nil , - 1 , nil
}
2018-03-06 06:52:16 +05:30
// GetAttachmentByID returns attachment by given id
func GetAttachmentByID ( id int64 ) ( * Attachment , error ) {
2021-09-19 17:19:59 +05:30
return getAttachmentByID ( db . DefaultContext ( ) . Engine ( ) , id )
2018-03-06 06:52:16 +05:30
}
2021-09-19 17:19:59 +05:30
func getAttachmentByID ( e db . Engine , id int64 ) ( * Attachment , error ) {
2020-06-17 23:20:11 +05:30
attach := & Attachment { }
if has , err := e . ID ( id ) . Get ( attach ) ; err != nil {
2018-03-06 06:52:16 +05:30
return nil , err
} else if ! has {
return nil , ErrAttachmentNotExist { ID : id , UUID : "" }
}
return attach , nil
}
2021-09-19 17:19:59 +05:30
func getAttachmentByUUID ( e db . Engine , uuid string ) ( * Attachment , error ) {
2020-06-17 23:20:11 +05:30
attach := & Attachment { }
has , err := e . Where ( "uuid=?" , uuid ) . Get ( attach )
2017-01-20 12:28:46 +05:30
if err != nil {
return nil , err
} else if ! has {
return nil , ErrAttachmentNotExist { 0 , uuid }
}
return attach , nil
}
2019-12-11 05:31:52 +05:30
// GetAttachmentsByUUIDs returns attachment by given UUID list.
2021-09-19 17:19:59 +05:30
func GetAttachmentsByUUIDs ( ctx * db . Context , uuids [ ] string ) ( [ ] * Attachment , error ) {
return getAttachmentsByUUIDs ( ctx . Engine ( ) , uuids )
2019-12-11 05:31:52 +05:30
}
2021-09-19 17:19:59 +05:30
func getAttachmentsByUUIDs ( e db . Engine , uuids [ ] string ) ( [ ] * Attachment , error ) {
2017-01-20 12:28:46 +05:30
if len ( uuids ) == 0 {
return [ ] * Attachment { } , nil
}
// Silently drop invalid uuids.
attachments := make ( [ ] * Attachment , 0 , len ( uuids ) )
return attachments , e . In ( "uuid" , uuids ) . Find ( & attachments )
}
// GetAttachmentByUUID returns attachment by given UUID.
func GetAttachmentByUUID ( uuid string ) ( * Attachment , error ) {
2021-09-19 17:19:59 +05:30
return getAttachmentByUUID ( db . DefaultContext ( ) . Engine ( ) , uuid )
2017-01-20 12:28:46 +05:30
}
2021-09-06 20:16:20 +05:30
// ExistAttachmentsByUUID returns true if attachment is exist by given UUID
func ExistAttachmentsByUUID ( uuid string ) ( bool , error ) {
2021-09-19 17:19:59 +05:30
return db . DefaultContext ( ) . Engine ( ) . Where ( "`uuid`=?" , uuid ) . Exist ( new ( Attachment ) )
2021-09-06 20:16:20 +05:30
}
2019-01-07 04:07:30 +05:30
// GetAttachmentByReleaseIDFileName returns attachment by given releaseId and fileName.
func GetAttachmentByReleaseIDFileName ( releaseID int64 , fileName string ) ( * Attachment , error ) {
2021-09-19 17:19:59 +05:30
return getAttachmentByReleaseIDFileName ( db . DefaultContext ( ) . Engine ( ) , releaseID , fileName )
2019-01-07 04:07:30 +05:30
}
2021-09-19 17:19:59 +05:30
func getAttachmentsByIssueID ( e db . Engine , issueID int64 ) ( [ ] * Attachment , error ) {
2017-01-20 12:28:46 +05:30
attachments := make ( [ ] * Attachment , 0 , 10 )
return attachments , e . Where ( "issue_id = ? AND comment_id = 0" , issueID ) . Find ( & attachments )
}
// GetAttachmentsByIssueID returns all attachments of an issue.
func GetAttachmentsByIssueID ( issueID int64 ) ( [ ] * Attachment , error ) {
2021-09-19 17:19:59 +05:30
return getAttachmentsByIssueID ( db . DefaultContext ( ) . Engine ( ) , issueID )
2017-01-20 12:28:46 +05:30
}
// GetAttachmentsByCommentID returns all attachments if comment by given ID.
func GetAttachmentsByCommentID ( commentID int64 ) ( [ ] * Attachment , error ) {
2021-09-19 17:19:59 +05:30
return getAttachmentsByCommentID ( db . DefaultContext ( ) . Engine ( ) , commentID )
2017-10-01 22:22:35 +05:30
}
2021-09-19 17:19:59 +05:30
func getAttachmentsByCommentID ( e db . Engine , commentID int64 ) ( [ ] * Attachment , error ) {
2017-01-20 12:28:46 +05:30
attachments := make ( [ ] * Attachment , 0 , 10 )
2020-02-28 04:40:27 +05:30
return attachments , e . Where ( "comment_id=?" , commentID ) . Find ( & attachments )
2017-01-20 12:28:46 +05:30
}
2019-01-07 04:07:30 +05:30
// getAttachmentByReleaseIDFileName return a file based on the the following infos:
2021-09-19 17:19:59 +05:30
func getAttachmentByReleaseIDFileName ( e db . Engine , releaseID int64 , fileName string ) ( * Attachment , error ) {
2019-01-07 04:07:30 +05:30
attach := & Attachment { ReleaseID : releaseID , Name : fileName }
has , err := e . Get ( attach )
if err != nil {
return nil , err
} else if ! has {
return nil , err
}
return attach , nil
}
2017-01-20 12:28:46 +05:30
// DeleteAttachment deletes the given attachment and optionally the associated file.
func DeleteAttachment ( a * Attachment , remove bool ) error {
2021-09-19 17:19:59 +05:30
_ , err := DeleteAttachments ( db . DefaultContext ( ) , [ ] * Attachment { a } , remove )
2017-01-20 12:28:46 +05:30
return err
}
// DeleteAttachments deletes the given attachments and optionally the associated files.
2021-09-19 17:19:59 +05:30
func DeleteAttachments ( ctx * db . Context , attachments [ ] * Attachment , remove bool ) ( int , error ) {
2017-12-25 02:34:22 +05:30
if len ( attachments ) == 0 {
return 0 , nil
}
2021-03-15 00:22:12 +05:30
ids := make ( [ ] int64 , 0 , len ( attachments ) )
2017-12-25 02:34:22 +05:30
for _ , a := range attachments {
ids = append ( ids , a . ID )
}
2021-09-19 17:19:59 +05:30
cnt , err := ctx . Engine ( ) . In ( "id" , ids ) . NoAutoCondition ( ) . Delete ( attachments [ 0 ] )
2017-12-25 02:34:22 +05:30
if err != nil {
return 0 , err
}
if remove {
for i , a := range attachments {
2020-08-18 09:53:45 +05:30
if err := storage . Attachments . Delete ( a . RelativePath ( ) ) ; err != nil {
2017-01-20 12:28:46 +05:30
return i , err
}
}
}
2017-12-25 02:34:22 +05:30
return int ( cnt ) , nil
2017-01-20 12:28:46 +05:30
}
// DeleteAttachmentsByIssue deletes all attachments associated with the given issue.
func DeleteAttachmentsByIssue ( issueID int64 , remove bool ) ( int , error ) {
attachments , err := GetAttachmentsByIssueID ( issueID )
if err != nil {
return 0 , err
}
2021-09-19 17:19:59 +05:30
return DeleteAttachments ( db . DefaultContext ( ) , attachments , remove )
2017-01-20 12:28:46 +05:30
}
// DeleteAttachmentsByComment deletes all attachments associated with the given comment.
func DeleteAttachmentsByComment ( commentID int64 , remove bool ) ( int , error ) {
attachments , err := GetAttachmentsByCommentID ( commentID )
if err != nil {
return 0 , err
}
2021-09-19 17:19:59 +05:30
return DeleteAttachments ( db . DefaultContext ( ) , attachments , remove )
2017-01-20 12:28:46 +05:30
}
2018-03-06 06:52:16 +05:30
// UpdateAttachment updates the given attachment in database
func UpdateAttachment ( atta * Attachment ) error {
2021-09-19 17:19:59 +05:30
return updateAttachment ( db . DefaultContext ( ) . Engine ( ) , atta )
2018-03-06 06:52:16 +05:30
}
2021-03-22 21:39:51 +05:30
// UpdateAttachmentByUUID Updates attachment via uuid
2021-09-19 17:19:59 +05:30
func UpdateAttachmentByUUID ( ctx * db . Context , attach * Attachment , cols ... string ) error {
2021-03-22 21:39:51 +05:30
if attach . UUID == "" {
return fmt . Errorf ( "Attachement uuid should not blank" )
}
2021-09-19 17:19:59 +05:30
_ , err := ctx . Engine ( ) . Where ( "uuid=?" , attach . UUID ) . Cols ( cols ... ) . Update ( attach )
2021-03-22 21:39:51 +05:30
return err
}
2021-09-19 17:19:59 +05:30
func updateAttachment ( e db . Engine , atta * Attachment ) error {
2018-03-06 06:52:16 +05:30
var sess * xorm . Session
if atta . ID != 0 && atta . UUID == "" {
sess = e . ID ( atta . ID )
} else {
// Use uuid only if id is not set and uuid is set
sess = e . Where ( "uuid = ?" , atta . UUID )
}
_ , err := sess . Cols ( "name" , "issue_id" , "release_id" , "comment_id" , "download_count" ) . Update ( atta )
return err
}
2019-09-30 21:40:00 +05:30
// DeleteAttachmentsByRelease deletes all attachments associated with the given release.
func DeleteAttachmentsByRelease ( releaseID int64 ) error {
2021-09-19 17:19:59 +05:30
_ , err := db . DefaultContext ( ) . Engine ( ) . Where ( "release_id = ?" , releaseID ) . Delete ( & Attachment { } )
2019-09-30 21:40:00 +05:30
return err
}
2020-08-18 09:53:45 +05:30
// IterateAttachment iterates attachments; it should not be used when Gitea is servicing users.
func IterateAttachment ( f func ( attach * Attachment ) error ) error {
var start int
const batchSize = 100
for {
2021-03-15 00:22:12 +05:30
attachments := make ( [ ] * Attachment , 0 , batchSize )
2021-09-19 17:19:59 +05:30
if err := db . DefaultContext ( ) . Engine ( ) . Limit ( batchSize , start ) . Find ( & attachments ) ; err != nil {
2020-08-18 09:53:45 +05:30
return err
}
if len ( attachments ) == 0 {
return nil
}
start += len ( attachments )
for _ , attach := range attachments {
if err := f ( attach ) ; err != nil {
return err
}
}
}
}
2021-09-15 01:11:40 +05:30
// CountOrphanedAttachments returns the number of bad attachments
func CountOrphanedAttachments ( ) ( int64 , error ) {
2021-09-19 17:19:59 +05:30
return db . DefaultContext ( ) . Engine ( ) . Where ( "(issue_id > 0 and issue_id not in (select id from issue)) or (release_id > 0 and release_id not in (select id from `release`))" ) .
2021-09-15 01:11:40 +05:30
Count ( new ( Attachment ) )
}
// DeleteOrphanedAttachments delete all bad attachments
func DeleteOrphanedAttachments ( ) error {
2021-09-19 17:19:59 +05:30
_ , err := db . DefaultContext ( ) . Engine ( ) . Where ( "(issue_id > 0 and issue_id not in (select id from issue)) or (release_id > 0 and release_id not in (select id from `release`))" ) .
2021-09-15 01:11:40 +05:30
Delete ( new ( Attachment ) )
return err
}