repo_collaboration.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. package models
  2. import (
  3. "fmt"
  4. api "gitlab.com/gitote/go-gitote-client"
  5. log "gopkg.in/clog.v1"
  6. )
  7. // Collaboration represent the relation between an individual and a repository.
  8. type Collaboration struct {
  9. ID int64
  10. RepoID int64 `xorm:"UNIQUE(s) INDEX NOT NULL"`
  11. UserID int64 `xorm:"UNIQUE(s) INDEX NOT NULL"`
  12. Mode AccessMode `xorm:"DEFAULT 2 NOT NULL"`
  13. }
  14. func (c *Collaboration) ModeI18nKey() string {
  15. switch c.Mode {
  16. case ACCESS_MODE_READ:
  17. return "repo.settings.collaboration.read"
  18. case ACCESS_MODE_WRITE:
  19. return "repo.settings.collaboration.write"
  20. case ACCESS_MODE_ADMIN:
  21. return "repo.settings.collaboration.admin"
  22. default:
  23. return "repo.settings.collaboration.undefined"
  24. }
  25. }
  26. // IsCollaborator returns true if the user is a collaborator of the repository.
  27. func IsCollaborator(repoID, userID int64) bool {
  28. collaboration := &Collaboration{
  29. RepoID: repoID,
  30. UserID: userID,
  31. }
  32. has, err := x.Get(collaboration)
  33. if err != nil {
  34. log.Error(2, "get collaboration [repo_id: %d, user_id: %d]: %v", repoID, userID, err)
  35. return false
  36. }
  37. return has
  38. }
  39. func (repo *Repository) IsCollaborator(userID int64) bool {
  40. return IsCollaborator(repo.ID, userID)
  41. }
  42. // AddCollaborator adds new collaboration to a repository with default access mode.
  43. func (repo *Repository) AddCollaborator(u *User) error {
  44. collaboration := &Collaboration{
  45. RepoID: repo.ID,
  46. UserID: u.ID,
  47. }
  48. has, err := x.Get(collaboration)
  49. if err != nil {
  50. return err
  51. } else if has {
  52. return nil
  53. }
  54. collaboration.Mode = ACCESS_MODE_WRITE
  55. sess := x.NewSession()
  56. defer sess.Close()
  57. if err = sess.Begin(); err != nil {
  58. return err
  59. }
  60. if _, err = sess.Insert(collaboration); err != nil {
  61. return err
  62. } else if err = repo.recalculateAccesses(sess); err != nil {
  63. return fmt.Errorf("recalculateAccesses [repo_id: %v]: %v", repo.ID, err)
  64. }
  65. return sess.Commit()
  66. }
  67. func (repo *Repository) getCollaborations(e Engine) ([]*Collaboration, error) {
  68. collaborations := make([]*Collaboration, 0)
  69. return collaborations, e.Find(&collaborations, &Collaboration{RepoID: repo.ID})
  70. }
  71. // Collaborator represents a user with collaboration details.
  72. type Collaborator struct {
  73. *User
  74. Collaboration *Collaboration
  75. }
  76. func (c *Collaborator) APIFormat() *api.Collaborator {
  77. return &api.Collaborator{
  78. User: c.User.APIFormat(),
  79. Permissions: api.Permission{
  80. Admin: c.Collaboration.Mode >= ACCESS_MODE_ADMIN,
  81. Push: c.Collaboration.Mode >= ACCESS_MODE_WRITE,
  82. Pull: c.Collaboration.Mode >= ACCESS_MODE_READ,
  83. },
  84. }
  85. }
  86. func (repo *Repository) getCollaborators(e Engine) ([]*Collaborator, error) {
  87. collaborations, err := repo.getCollaborations(e)
  88. if err != nil {
  89. return nil, fmt.Errorf("getCollaborations: %v", err)
  90. }
  91. collaborators := make([]*Collaborator, len(collaborations))
  92. for i, c := range collaborations {
  93. user, err := getUserByID(e, c.UserID)
  94. if err != nil {
  95. return nil, err
  96. }
  97. collaborators[i] = &Collaborator{
  98. User: user,
  99. Collaboration: c,
  100. }
  101. }
  102. return collaborators, nil
  103. }
  104. // GetCollaborators returns the collaborators for a repository
  105. func (repo *Repository) GetCollaborators() ([]*Collaborator, error) {
  106. return repo.getCollaborators(x)
  107. }
  108. // ChangeCollaborationAccessMode sets new access mode for the collaboration.
  109. func (repo *Repository) ChangeCollaborationAccessMode(userID int64, mode AccessMode) error {
  110. // Discard invalid input
  111. if mode <= ACCESS_MODE_NONE || mode > ACCESS_MODE_OWNER {
  112. return nil
  113. }
  114. collaboration := &Collaboration{
  115. RepoID: repo.ID,
  116. UserID: userID,
  117. }
  118. has, err := x.Get(collaboration)
  119. if err != nil {
  120. return fmt.Errorf("get collaboration: %v", err)
  121. } else if !has {
  122. return nil
  123. }
  124. if collaboration.Mode == mode {
  125. return nil
  126. }
  127. collaboration.Mode = mode
  128. // If it's an organizational repository, merge with team access level for highest permission
  129. if repo.Owner.IsOrganization() {
  130. teams, err := GetUserTeams(repo.OwnerID, userID)
  131. if err != nil {
  132. return fmt.Errorf("GetUserTeams: [org_id: %d, user_id: %d]: %v", repo.OwnerID, userID, err)
  133. }
  134. for i := range teams {
  135. if mode < teams[i].Authorize {
  136. mode = teams[i].Authorize
  137. }
  138. }
  139. }
  140. sess := x.NewSession()
  141. defer sess.Close()
  142. if err = sess.Begin(); err != nil {
  143. return err
  144. }
  145. if _, err = sess.ID(collaboration.ID).AllCols().Update(collaboration); err != nil {
  146. return fmt.Errorf("update collaboration: %v", err)
  147. }
  148. access := &Access{
  149. UserID: userID,
  150. RepoID: repo.ID,
  151. }
  152. has, err = sess.Get(access)
  153. if err != nil {
  154. return fmt.Errorf("get access record: %v", err)
  155. }
  156. if has {
  157. _, err = sess.Exec("UPDATE access SET mode = ? WHERE user_id = ? AND repo_id = ?", mode, userID, repo.ID)
  158. } else {
  159. access.Mode = mode
  160. _, err = sess.Insert(access)
  161. }
  162. if err != nil {
  163. return fmt.Errorf("update/insert access table: %v", err)
  164. }
  165. return sess.Commit()
  166. }
  167. // DeleteCollaboration removes collaboration relation between the user and repository.
  168. func DeleteCollaboration(repo *Repository, userID int64) (err error) {
  169. if !IsCollaborator(repo.ID, userID) {
  170. return nil
  171. }
  172. collaboration := &Collaboration{
  173. RepoID: repo.ID,
  174. UserID: userID,
  175. }
  176. sess := x.NewSession()
  177. defer sess.Close()
  178. if err = sess.Begin(); err != nil {
  179. return err
  180. }
  181. if has, err := sess.Delete(collaboration); err != nil || has == 0 {
  182. return err
  183. } else if err = repo.recalculateAccesses(sess); err != nil {
  184. return err
  185. }
  186. return sess.Commit()
  187. }
  188. func (repo *Repository) DeleteCollaboration(userID int64) error {
  189. return DeleteCollaboration(repo, userID)
  190. }