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.

lfs.go 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437
  1. // Copyright 2020 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package git
  4. import (
  5. "context"
  6. "fmt"
  7. "code.gitea.io/gitea/models/db"
  8. "code.gitea.io/gitea/models/perm"
  9. repo_model "code.gitea.io/gitea/models/repo"
  10. "code.gitea.io/gitea/models/unit"
  11. user_model "code.gitea.io/gitea/models/user"
  12. "code.gitea.io/gitea/modules/lfs"
  13. "code.gitea.io/gitea/modules/log"
  14. "code.gitea.io/gitea/modules/setting"
  15. "code.gitea.io/gitea/modules/timeutil"
  16. "code.gitea.io/gitea/modules/util"
  17. "xorm.io/builder"
  18. )
  19. // ErrLFSLockNotExist represents a "LFSLockNotExist" kind of error.
  20. type ErrLFSLockNotExist struct {
  21. ID int64
  22. RepoID int64
  23. Path string
  24. }
  25. // IsErrLFSLockNotExist checks if an error is a ErrLFSLockNotExist.
  26. func IsErrLFSLockNotExist(err error) bool {
  27. _, ok := err.(ErrLFSLockNotExist)
  28. return ok
  29. }
  30. func (err ErrLFSLockNotExist) Error() string {
  31. return fmt.Sprintf("lfs lock does not exist [id: %d, rid: %d, path: %s]", err.ID, err.RepoID, err.Path)
  32. }
  33. func (err ErrLFSLockNotExist) Unwrap() error {
  34. return util.ErrNotExist
  35. }
  36. // ErrLFSUnauthorizedAction represents a "LFSUnauthorizedAction" kind of error.
  37. type ErrLFSUnauthorizedAction struct {
  38. RepoID int64
  39. UserName string
  40. Mode perm.AccessMode
  41. }
  42. // IsErrLFSUnauthorizedAction checks if an error is a ErrLFSUnauthorizedAction.
  43. func IsErrLFSUnauthorizedAction(err error) bool {
  44. _, ok := err.(ErrLFSUnauthorizedAction)
  45. return ok
  46. }
  47. func (err ErrLFSUnauthorizedAction) Error() string {
  48. if err.Mode == perm.AccessModeWrite {
  49. return fmt.Sprintf("User %s doesn't have write access for lfs lock [rid: %d]", err.UserName, err.RepoID)
  50. }
  51. return fmt.Sprintf("User %s doesn't have read access for lfs lock [rid: %d]", err.UserName, err.RepoID)
  52. }
  53. func (err ErrLFSUnauthorizedAction) Unwrap() error {
  54. return util.ErrPermissionDenied
  55. }
  56. // ErrLFSLockAlreadyExist represents a "LFSLockAlreadyExist" kind of error.
  57. type ErrLFSLockAlreadyExist struct {
  58. RepoID int64
  59. Path string
  60. }
  61. // IsErrLFSLockAlreadyExist checks if an error is a ErrLFSLockAlreadyExist.
  62. func IsErrLFSLockAlreadyExist(err error) bool {
  63. _, ok := err.(ErrLFSLockAlreadyExist)
  64. return ok
  65. }
  66. func (err ErrLFSLockAlreadyExist) Error() string {
  67. return fmt.Sprintf("lfs lock already exists [rid: %d, path: %s]", err.RepoID, err.Path)
  68. }
  69. func (err ErrLFSLockAlreadyExist) Unwrap() error {
  70. return util.ErrAlreadyExist
  71. }
  72. // ErrLFSFileLocked represents a "LFSFileLocked" kind of error.
  73. type ErrLFSFileLocked struct {
  74. RepoID int64
  75. Path string
  76. UserName string
  77. }
  78. // IsErrLFSFileLocked checks if an error is a ErrLFSFileLocked.
  79. func IsErrLFSFileLocked(err error) bool {
  80. _, ok := err.(ErrLFSFileLocked)
  81. return ok
  82. }
  83. func (err ErrLFSFileLocked) Error() string {
  84. return fmt.Sprintf("File is lfs locked [repo: %d, locked by: %s, path: %s]", err.RepoID, err.UserName, err.Path)
  85. }
  86. func (err ErrLFSFileLocked) Unwrap() error {
  87. return util.ErrPermissionDenied
  88. }
  89. // LFSMetaObject stores metadata for LFS tracked files.
  90. type LFSMetaObject struct {
  91. ID int64 `xorm:"pk autoincr"`
  92. lfs.Pointer `xorm:"extends"`
  93. RepositoryID int64 `xorm:"UNIQUE(s) INDEX NOT NULL"`
  94. Existing bool `xorm:"-"`
  95. CreatedUnix timeutil.TimeStamp `xorm:"created"`
  96. UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"`
  97. }
  98. func init() {
  99. db.RegisterModel(new(LFSMetaObject))
  100. }
  101. // LFSTokenResponse defines the JSON structure in which the JWT token is stored.
  102. // This structure is fetched via SSH and passed by the Git LFS client to the server
  103. // endpoint for authorization.
  104. type LFSTokenResponse struct {
  105. Header map[string]string `json:"header"`
  106. Href string `json:"href"`
  107. }
  108. // ErrLFSObjectNotExist is returned from lfs models functions in order
  109. // to differentiate between database and missing object errors.
  110. var ErrLFSObjectNotExist = db.ErrNotExist{Resource: "LFS Meta object"}
  111. // NewLFSMetaObject stores a given populated LFSMetaObject structure in the database
  112. // if it is not already present.
  113. func NewLFSMetaObject(ctx context.Context, m *LFSMetaObject) (*LFSMetaObject, error) {
  114. var err error
  115. ctx, committer, err := db.TxContext(ctx)
  116. if err != nil {
  117. return nil, err
  118. }
  119. defer committer.Close()
  120. has, err := db.GetByBean(ctx, m)
  121. if err != nil {
  122. return nil, err
  123. }
  124. if has {
  125. m.Existing = true
  126. return m, committer.Commit()
  127. }
  128. if err = db.Insert(ctx, m); err != nil {
  129. return nil, err
  130. }
  131. return m, committer.Commit()
  132. }
  133. // GetLFSMetaObjectByOid selects a LFSMetaObject entry from database by its OID.
  134. // It may return ErrLFSObjectNotExist or a database error. If the error is nil,
  135. // the returned pointer is a valid LFSMetaObject.
  136. func GetLFSMetaObjectByOid(ctx context.Context, repoID int64, oid string) (*LFSMetaObject, error) {
  137. if len(oid) == 0 {
  138. return nil, ErrLFSObjectNotExist
  139. }
  140. m := &LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}, RepositoryID: repoID}
  141. has, err := db.GetEngine(ctx).Get(m)
  142. if err != nil {
  143. return nil, err
  144. } else if !has {
  145. return nil, ErrLFSObjectNotExist
  146. }
  147. return m, nil
  148. }
  149. // RemoveLFSMetaObjectByOid removes a LFSMetaObject entry from database by its OID.
  150. // It may return ErrLFSObjectNotExist or a database error.
  151. func RemoveLFSMetaObjectByOid(ctx context.Context, repoID int64, oid string) (int64, error) {
  152. return RemoveLFSMetaObjectByOidFn(ctx, repoID, oid, nil)
  153. }
  154. // RemoveLFSMetaObjectByOidFn removes a LFSMetaObject entry from database by its OID.
  155. // It may return ErrLFSObjectNotExist or a database error. It will run Fn with the current count within the transaction
  156. func RemoveLFSMetaObjectByOidFn(ctx context.Context, repoID int64, oid string, fn func(count int64) error) (int64, error) {
  157. if len(oid) == 0 {
  158. return 0, ErrLFSObjectNotExist
  159. }
  160. ctx, committer, err := db.TxContext(ctx)
  161. if err != nil {
  162. return 0, err
  163. }
  164. defer committer.Close()
  165. m := &LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}, RepositoryID: repoID}
  166. if _, err := db.DeleteByBean(ctx, m); err != nil {
  167. return -1, err
  168. }
  169. count, err := db.CountByBean(ctx, &LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}})
  170. if err != nil {
  171. return count, err
  172. }
  173. if fn != nil {
  174. if err := fn(count); err != nil {
  175. return count, err
  176. }
  177. }
  178. return count, committer.Commit()
  179. }
  180. // GetLFSMetaObjects returns all LFSMetaObjects associated with a repository
  181. func GetLFSMetaObjects(ctx context.Context, repoID int64, page, pageSize int) ([]*LFSMetaObject, error) {
  182. sess := db.GetEngine(ctx)
  183. if page >= 0 && pageSize > 0 {
  184. start := 0
  185. if page > 0 {
  186. start = (page - 1) * pageSize
  187. }
  188. sess.Limit(pageSize, start)
  189. }
  190. lfsObjects := make([]*LFSMetaObject, 0, pageSize)
  191. return lfsObjects, sess.Find(&lfsObjects, &LFSMetaObject{RepositoryID: repoID})
  192. }
  193. // CountLFSMetaObjects returns a count of all LFSMetaObjects associated with a repository
  194. func CountLFSMetaObjects(ctx context.Context, repoID int64) (int64, error) {
  195. return db.GetEngine(ctx).Count(&LFSMetaObject{RepositoryID: repoID})
  196. }
  197. // LFSObjectAccessible checks if a provided Oid is accessible to the user
  198. func LFSObjectAccessible(ctx context.Context, user *user_model.User, oid string) (bool, error) {
  199. if user.IsAdmin {
  200. count, err := db.GetEngine(ctx).Count(&LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}})
  201. return count > 0, err
  202. }
  203. cond := repo_model.AccessibleRepositoryCondition(user, unit.TypeInvalid)
  204. count, err := db.GetEngine(ctx).Where(cond).Join("INNER", "repository", "`lfs_meta_object`.repository_id = `repository`.id").Count(&LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}})
  205. return count > 0, err
  206. }
  207. // ExistsLFSObject checks if a provided Oid exists within the DB
  208. func ExistsLFSObject(ctx context.Context, oid string) (bool, error) {
  209. return db.GetEngine(ctx).Exist(&LFSMetaObject{Pointer: lfs.Pointer{Oid: oid}})
  210. }
  211. // LFSAutoAssociate auto associates accessible LFSMetaObjects
  212. func LFSAutoAssociate(ctx context.Context, metas []*LFSMetaObject, user *user_model.User, repoID int64) error {
  213. ctx, committer, err := db.TxContext(ctx)
  214. if err != nil {
  215. return err
  216. }
  217. defer committer.Close()
  218. sess := db.GetEngine(ctx)
  219. oids := make([]any, len(metas))
  220. oidMap := make(map[string]*LFSMetaObject, len(metas))
  221. for i, meta := range metas {
  222. oids[i] = meta.Oid
  223. oidMap[meta.Oid] = meta
  224. }
  225. if !user.IsAdmin {
  226. newMetas := make([]*LFSMetaObject, 0, len(metas))
  227. cond := builder.In(
  228. "`lfs_meta_object`.repository_id",
  229. builder.Select("`repository`.id").From("repository").Where(repo_model.AccessibleRepositoryCondition(user, unit.TypeInvalid)),
  230. )
  231. err = sess.Cols("oid").Where(cond).In("oid", oids...).GroupBy("oid").Find(&newMetas)
  232. if err != nil {
  233. return err
  234. }
  235. if len(newMetas) != len(oidMap) {
  236. return fmt.Errorf("unable collect all LFS objects from database, expected %d, actually %d", len(oidMap), len(newMetas))
  237. }
  238. for i := range newMetas {
  239. newMetas[i].Size = oidMap[newMetas[i].Oid].Size
  240. newMetas[i].RepositoryID = repoID
  241. }
  242. if err = db.Insert(ctx, newMetas); err != nil {
  243. return err
  244. }
  245. } else {
  246. // admin can associate any LFS object to any repository, and we do not care about errors (eg: duplicated unique key),
  247. // even if error occurs, it won't hurt users and won't make things worse
  248. for i := range metas {
  249. p := lfs.Pointer{Oid: metas[i].Oid, Size: metas[i].Size}
  250. _, err = sess.Insert(&LFSMetaObject{
  251. Pointer: p,
  252. RepositoryID: repoID,
  253. })
  254. if err != nil {
  255. log.Warn("failed to insert LFS meta object %-v for repo_id: %d into database, err=%v", p, repoID, err)
  256. }
  257. }
  258. }
  259. return committer.Commit()
  260. }
  261. // CopyLFS copies LFS data from one repo to another
  262. func CopyLFS(ctx context.Context, newRepo, oldRepo *repo_model.Repository) error {
  263. var lfsObjects []*LFSMetaObject
  264. if err := db.GetEngine(ctx).Where("repository_id=?", oldRepo.ID).Find(&lfsObjects); err != nil {
  265. return err
  266. }
  267. for _, v := range lfsObjects {
  268. v.ID = 0
  269. v.RepositoryID = newRepo.ID
  270. if err := db.Insert(ctx, v); err != nil {
  271. return err
  272. }
  273. }
  274. return nil
  275. }
  276. // GetRepoLFSSize return a repository's lfs files size
  277. func GetRepoLFSSize(ctx context.Context, repoID int64) (int64, error) {
  278. lfsSize, err := db.GetEngine(ctx).Where("repository_id = ?", repoID).SumInt(new(LFSMetaObject), "size")
  279. if err != nil {
  280. return 0, fmt.Errorf("updateSize: GetLFSMetaObjects: %w", err)
  281. }
  282. return lfsSize, nil
  283. }
  284. // IterateRepositoryIDsWithLFSMetaObjects iterates across the repositories that have LFSMetaObjects
  285. func IterateRepositoryIDsWithLFSMetaObjects(ctx context.Context, f func(ctx context.Context, repoID, count int64) error) error {
  286. batchSize := setting.Database.IterateBufferSize
  287. sess := db.GetEngine(ctx)
  288. id := int64(0)
  289. type RepositoryCount struct {
  290. RepositoryID int64
  291. Count int64
  292. }
  293. for {
  294. counts := make([]*RepositoryCount, 0, batchSize)
  295. sess.Select("repository_id, COUNT(id) AS count").
  296. Table("lfs_meta_object").
  297. Where("repository_id > ?", id).
  298. GroupBy("repository_id").
  299. OrderBy("repository_id ASC")
  300. if err := sess.Limit(batchSize, 0).Find(&counts); err != nil {
  301. return err
  302. }
  303. if len(counts) == 0 {
  304. return nil
  305. }
  306. for _, count := range counts {
  307. if err := f(ctx, count.RepositoryID, count.Count); err != nil {
  308. return err
  309. }
  310. }
  311. id = counts[len(counts)-1].RepositoryID
  312. }
  313. }
  314. // IterateLFSMetaObjectsForRepoOptions provides options for IterateLFSMetaObjectsForRepo
  315. type IterateLFSMetaObjectsForRepoOptions struct {
  316. OlderThan timeutil.TimeStamp
  317. UpdatedLessRecentlyThan timeutil.TimeStamp
  318. OrderByUpdated bool
  319. LoopFunctionAlwaysUpdates bool
  320. }
  321. // IterateLFSMetaObjectsForRepo provides a iterator for LFSMetaObjects per Repo
  322. func IterateLFSMetaObjectsForRepo(ctx context.Context, repoID int64, f func(context.Context, *LFSMetaObject, int64) error, opts *IterateLFSMetaObjectsForRepoOptions) error {
  323. var start int
  324. batchSize := setting.Database.IterateBufferSize
  325. engine := db.GetEngine(ctx)
  326. type CountLFSMetaObject struct {
  327. Count int64
  328. LFSMetaObject `xorm:"extends"`
  329. }
  330. id := int64(0)
  331. for {
  332. beans := make([]*CountLFSMetaObject, 0, batchSize)
  333. sess := engine.Table("lfs_meta_object").Select("`lfs_meta_object`.*, COUNT(`l1`.oid) AS `count`").
  334. Join("INNER", "`lfs_meta_object` AS l1", "`lfs_meta_object`.oid = `l1`.oid").
  335. Where("`lfs_meta_object`.repository_id = ?", repoID)
  336. if !opts.OlderThan.IsZero() {
  337. sess.And("`lfs_meta_object`.created_unix < ?", opts.OlderThan)
  338. }
  339. if !opts.UpdatedLessRecentlyThan.IsZero() {
  340. sess.And("`lfs_meta_object`.updated_unix < ?", opts.UpdatedLessRecentlyThan)
  341. }
  342. sess.GroupBy("`lfs_meta_object`.id")
  343. if opts.OrderByUpdated {
  344. sess.OrderBy("`lfs_meta_object`.updated_unix ASC")
  345. } else {
  346. sess.And("`lfs_meta_object`.id > ?", id)
  347. sess.OrderBy("`lfs_meta_object`.id ASC")
  348. }
  349. if err := sess.Limit(batchSize, start).Find(&beans); err != nil {
  350. return err
  351. }
  352. if len(beans) == 0 {
  353. return nil
  354. }
  355. if !opts.LoopFunctionAlwaysUpdates {
  356. start += len(beans)
  357. }
  358. for _, bean := range beans {
  359. if err := f(ctx, &bean.LFSMetaObject, bean.Count); err != nil {
  360. return err
  361. }
  362. }
  363. id = beans[len(beans)-1].ID
  364. }
  365. }
  366. // MarkLFSMetaObject updates the updated time for the provided LFSMetaObject
  367. func MarkLFSMetaObject(ctx context.Context, id int64) error {
  368. obj := &LFSMetaObject{
  369. UpdatedUnix: timeutil.TimeStampNow(),
  370. }
  371. count, err := db.GetEngine(ctx).ID(id).Update(obj)
  372. if count != 1 {
  373. log.Error("Unexpectedly updated %d LFSMetaObjects with ID: %d", count, id)
  374. }
  375. return err
  376. }