perms.go 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. // Copyright 2020 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package db
  5. import (
  6. "gorm.io/gorm"
  7. log "unknwon.dev/clog/v2"
  8. )
  9. // PermsStore is the persistent interface for permissions.
  10. //
  11. // NOTE: All methods are sorted in alphabetical order.
  12. type PermsStore interface {
  13. // AccessMode returns the access mode of given user has to the repository.
  14. AccessMode(userID int64, repo *Repository) AccessMode
  15. // Authorize returns true if the user has as good as desired access mode to the repository.
  16. Authorize(userID int64, repo *Repository, desired AccessMode) bool
  17. // SetRepoPerms does a full update to which users have which level of access to given repository.
  18. // Keys of the "accessMap" are user IDs.
  19. SetRepoPerms(repoID int64, accessMap map[int64]AccessMode) error
  20. }
  21. var Perms PermsStore
  22. var _ PermsStore = (*perms)(nil)
  23. type perms struct {
  24. *gorm.DB
  25. }
  26. func (db *perms) AccessMode(userID int64, repo *Repository) (mode AccessMode) {
  27. if repo == nil {
  28. return AccessModeNone
  29. }
  30. // Everyone has read access to public repository.
  31. if !repo.IsPrivate {
  32. mode = AccessModeRead
  33. }
  34. // Anonymous user gets the default access.
  35. if userID <= 0 {
  36. return mode
  37. }
  38. if userID == repo.OwnerID {
  39. return AccessModeOwner
  40. }
  41. access := new(Access)
  42. err := db.Where("user_id = ? AND repo_id = ?", userID, repo.ID).First(access).Error
  43. if err != nil {
  44. if err != gorm.ErrRecordNotFound {
  45. log.Error("Failed to get access [user_id: %d, repo_id: %d]: %v", userID, repo.ID, err)
  46. }
  47. return mode
  48. }
  49. return access.Mode
  50. }
  51. func (db *perms) Authorize(userID int64, repo *Repository, desired AccessMode) bool {
  52. return desired <= db.AccessMode(userID, repo)
  53. }
  54. func (db *perms) SetRepoPerms(repoID int64, accessMap map[int64]AccessMode) error {
  55. records := make([]*Access, 0, len(accessMap))
  56. for userID, mode := range accessMap {
  57. records = append(records, &Access{
  58. UserID: userID,
  59. RepoID: repoID,
  60. Mode: mode,
  61. })
  62. }
  63. return db.Transaction(func(tx *gorm.DB) error {
  64. err := tx.Where("repo_id = ?", repoID).Delete(new(Access)).Error
  65. if err != nil {
  66. return err
  67. }
  68. return tx.Create(&records).Error
  69. })
  70. }