You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

upload.go 4.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. // Copyright 2016 The Gogs Authors. All rights reserved.
  2. // Copyright 2019 The Gitea Authors. All rights reserved.
  3. // SPDX-License-Identifier: MIT
  4. package repo
  5. import (
  6. "fmt"
  7. "io"
  8. "mime/multipart"
  9. "os"
  10. "path"
  11. "code.gitea.io/gitea/models/db"
  12. "code.gitea.io/gitea/modules/log"
  13. "code.gitea.io/gitea/modules/setting"
  14. "code.gitea.io/gitea/modules/util"
  15. gouuid "github.com/google/uuid"
  16. )
  17. // ErrUploadNotExist represents a "UploadNotExist" kind of error.
  18. type ErrUploadNotExist struct {
  19. ID int64
  20. UUID string
  21. }
  22. // IsErrUploadNotExist checks if an error is a ErrUploadNotExist.
  23. func IsErrUploadNotExist(err error) bool {
  24. _, ok := err.(ErrUploadNotExist)
  25. return ok
  26. }
  27. func (err ErrUploadNotExist) Error() string {
  28. return fmt.Sprintf("attachment does not exist [id: %d, uuid: %s]", err.ID, err.UUID)
  29. }
  30. func (err ErrUploadNotExist) Unwrap() error {
  31. return util.ErrNotExist
  32. }
  33. // Upload represent a uploaded file to a repo to be deleted when moved
  34. type Upload struct {
  35. ID int64 `xorm:"pk autoincr"`
  36. UUID string `xorm:"uuid UNIQUE"`
  37. Name string
  38. }
  39. func init() {
  40. db.RegisterModel(new(Upload))
  41. }
  42. // UploadLocalPath returns where uploads is stored in local file system based on given UUID.
  43. func UploadLocalPath(uuid string) string {
  44. return path.Join(setting.Repository.Upload.TempPath, uuid[0:1], uuid[1:2], uuid)
  45. }
  46. // LocalPath returns where uploads are temporarily stored in local file system.
  47. func (upload *Upload) LocalPath() string {
  48. return UploadLocalPath(upload.UUID)
  49. }
  50. // NewUpload creates a new upload object.
  51. func NewUpload(name string, buf []byte, file multipart.File) (_ *Upload, err error) {
  52. upload := &Upload{
  53. UUID: gouuid.New().String(),
  54. Name: name,
  55. }
  56. localPath := upload.LocalPath()
  57. if err = os.MkdirAll(path.Dir(localPath), os.ModePerm); err != nil {
  58. return nil, fmt.Errorf("MkdirAll: %w", err)
  59. }
  60. fw, err := os.Create(localPath)
  61. if err != nil {
  62. return nil, fmt.Errorf("Create: %w", err)
  63. }
  64. defer fw.Close()
  65. if _, err = fw.Write(buf); err != nil {
  66. return nil, fmt.Errorf("Write: %w", err)
  67. } else if _, err = io.Copy(fw, file); err != nil {
  68. return nil, fmt.Errorf("Copy: %w", err)
  69. }
  70. if _, err := db.GetEngine(db.DefaultContext).Insert(upload); err != nil {
  71. return nil, err
  72. }
  73. return upload, nil
  74. }
  75. // GetUploadByUUID returns the Upload by UUID
  76. func GetUploadByUUID(uuid string) (*Upload, error) {
  77. upload := &Upload{}
  78. has, err := db.GetEngine(db.DefaultContext).Where("uuid=?", uuid).Get(upload)
  79. if err != nil {
  80. return nil, err
  81. } else if !has {
  82. return nil, ErrUploadNotExist{0, uuid}
  83. }
  84. return upload, nil
  85. }
  86. // GetUploadsByUUIDs returns multiple uploads by UUIDS
  87. func GetUploadsByUUIDs(uuids []string) ([]*Upload, error) {
  88. if len(uuids) == 0 {
  89. return []*Upload{}, nil
  90. }
  91. // Silently drop invalid uuids.
  92. uploads := make([]*Upload, 0, len(uuids))
  93. return uploads, db.GetEngine(db.DefaultContext).In("uuid", uuids).Find(&uploads)
  94. }
  95. // DeleteUploads deletes multiple uploads
  96. func DeleteUploads(uploads ...*Upload) (err error) {
  97. if len(uploads) == 0 {
  98. return nil
  99. }
  100. ctx, committer, err := db.TxContext(db.DefaultContext)
  101. if err != nil {
  102. return err
  103. }
  104. defer committer.Close()
  105. ids := make([]int64, len(uploads))
  106. for i := 0; i < len(uploads); i++ {
  107. ids[i] = uploads[i].ID
  108. }
  109. if _, err = db.GetEngine(ctx).
  110. In("id", ids).
  111. Delete(new(Upload)); err != nil {
  112. return fmt.Errorf("delete uploads: %w", err)
  113. }
  114. if err = committer.Commit(); err != nil {
  115. return err
  116. }
  117. for _, upload := range uploads {
  118. localPath := upload.LocalPath()
  119. isFile, err := util.IsFile(localPath)
  120. if err != nil {
  121. log.Error("Unable to check if %s is a file. Error: %v", localPath, err)
  122. }
  123. if !isFile {
  124. continue
  125. }
  126. if err := util.Remove(localPath); err != nil {
  127. return fmt.Errorf("remove upload: %w", err)
  128. }
  129. }
  130. return nil
  131. }
  132. // DeleteUploadByUUID deletes a upload by UUID
  133. func DeleteUploadByUUID(uuid string) error {
  134. upload, err := GetUploadByUUID(uuid)
  135. if err != nil {
  136. if IsErrUploadNotExist(err) {
  137. return nil
  138. }
  139. return fmt.Errorf("GetUploadByUUID: %w", err)
  140. }
  141. if err := DeleteUploads(upload); err != nil {
  142. return fmt.Errorf("DeleteUpload: %w", err)
  143. }
  144. return nil
  145. }