permissions.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. package database
  2. import (
  3. "context"
  4. "github.com/cockroachdb/errors"
  5. "gorm.io/gorm"
  6. log "unknwon.dev/clog/v2"
  7. )
  8. // Access represents the highest access level of a user has to a repository. The
  9. // only access type that is not in this table is the real owner of a repository.
  10. // In case of an organization repository, the members of the owners team are in
  11. // this table.
  12. type Access struct {
  13. ID int64 `gorm:"primaryKey"`
  14. UserID int64 `xorm:"UNIQUE(s)" gorm:"uniqueIndex:access_user_repo_unique;not null"`
  15. RepoID int64 `xorm:"UNIQUE(s)" gorm:"uniqueIndex:access_user_repo_unique;not null"`
  16. Mode AccessMode `gorm:"not null"`
  17. }
  18. // AccessMode is the access mode of a user has to a repository.
  19. type AccessMode int
  20. const (
  21. AccessModeNone AccessMode = iota // 0
  22. AccessModeRead // 1
  23. AccessModeWrite // 2
  24. AccessModeAdmin // 3
  25. AccessModeOwner // 4
  26. )
  27. func (mode AccessMode) String() string {
  28. switch mode {
  29. case AccessModeRead:
  30. return "read"
  31. case AccessModeWrite:
  32. return "write"
  33. case AccessModeAdmin:
  34. return "admin"
  35. case AccessModeOwner:
  36. return "owner"
  37. default:
  38. return "none"
  39. }
  40. }
  41. // ParseAccessMode returns corresponding access mode to given permission string.
  42. func ParseAccessMode(permission string) AccessMode {
  43. switch permission {
  44. case "write":
  45. return AccessModeWrite
  46. case "admin":
  47. return AccessModeAdmin
  48. default:
  49. return AccessModeRead
  50. }
  51. }
  52. // PermissionsStore is the storage layer for repository permissions.
  53. type PermissionsStore struct {
  54. db *gorm.DB
  55. }
  56. func newPermissionsStore(db *gorm.DB) *PermissionsStore {
  57. return &PermissionsStore{db: db}
  58. }
  59. type AccessModeOptions struct {
  60. OwnerID int64 // The ID of the repository owner.
  61. Private bool // Whether the repository is private.
  62. }
  63. // AccessMode returns the access mode of given user has to the repository.
  64. func (s *PermissionsStore) AccessMode(ctx context.Context, userID, repoID int64, opts AccessModeOptions) (mode AccessMode) {
  65. if repoID <= 0 {
  66. return AccessModeNone
  67. }
  68. // Everyone has read access to public repository.
  69. if !opts.Private {
  70. mode = AccessModeRead
  71. }
  72. // Anonymous user gets the default access.
  73. if userID <= 0 {
  74. return mode
  75. }
  76. if userID == opts.OwnerID {
  77. return AccessModeOwner
  78. }
  79. access := new(Access)
  80. err := s.db.WithContext(ctx).Where("user_id = ? AND repo_id = ?", userID, repoID).First(access).Error
  81. if err != nil {
  82. if !errors.Is(err, gorm.ErrRecordNotFound) {
  83. log.Error("Failed to get access [user_id: %d, repo_id: %d]: %v", userID, repoID, err)
  84. }
  85. return mode
  86. }
  87. return access.Mode
  88. }
  89. // Authorize returns true if the user has as good as desired access mode to the
  90. // repository.
  91. func (s *PermissionsStore) Authorize(ctx context.Context, userID, repoID int64, desired AccessMode, opts AccessModeOptions) bool {
  92. return desired <= s.AccessMode(ctx, userID, repoID, opts)
  93. }
  94. // SetRepoPerms does a full update to which users have which level of access to
  95. // given repository. Keys of the "accessMap" are user IDs.
  96. func (s *PermissionsStore) SetRepoPerms(ctx context.Context, repoID int64, accessMap map[int64]AccessMode) error {
  97. records := make([]*Access, 0, len(accessMap))
  98. for userID, mode := range accessMap {
  99. records = append(records, &Access{
  100. UserID: userID,
  101. RepoID: repoID,
  102. Mode: mode,
  103. })
  104. }
  105. return s.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  106. err := tx.Where("repo_id = ?", repoID).Delete(new(Access)).Error
  107. if err != nil {
  108. return err
  109. }
  110. return tx.Create(&records).Error
  111. })
  112. }