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.

members.go 3.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Copyright 2020 The Gitea Authors.
  3. // SPDX-License-Identifier: MIT
  4. package org
  5. import (
  6. "net/http"
  7. "code.gitea.io/gitea/models"
  8. "code.gitea.io/gitea/models/organization"
  9. "code.gitea.io/gitea/modules/base"
  10. "code.gitea.io/gitea/modules/context"
  11. "code.gitea.io/gitea/modules/log"
  12. "code.gitea.io/gitea/modules/setting"
  13. shared_user "code.gitea.io/gitea/routers/web/shared/user"
  14. )
  15. const (
  16. // tplMembers template for organization members page
  17. tplMembers base.TplName = "org/member/members"
  18. )
  19. // Members render organization users page
  20. func Members(ctx *context.Context) {
  21. org := ctx.Org.Organization
  22. ctx.Data["Title"] = org.FullName
  23. ctx.Data["PageIsOrgMembers"] = true
  24. page := ctx.FormInt("page")
  25. if page <= 1 {
  26. page = 1
  27. }
  28. opts := &organization.FindOrgMembersOpts{
  29. OrgID: org.ID,
  30. PublicOnly: true,
  31. }
  32. if ctx.Doer != nil {
  33. isMember, err := ctx.Org.Organization.IsOrgMember(ctx, ctx.Doer.ID)
  34. if err != nil {
  35. ctx.Error(http.StatusInternalServerError, "IsOrgMember")
  36. return
  37. }
  38. opts.PublicOnly = !isMember && !ctx.Doer.IsAdmin
  39. }
  40. ctx.Data["PublicOnly"] = opts.PublicOnly
  41. total, err := organization.CountOrgMembers(ctx, opts)
  42. if err != nil {
  43. ctx.Error(http.StatusInternalServerError, "CountOrgMembers")
  44. return
  45. }
  46. err = shared_user.LoadHeaderCount(ctx)
  47. if err != nil {
  48. ctx.ServerError("LoadHeaderCount", err)
  49. return
  50. }
  51. pager := context.NewPagination(int(total), setting.UI.MembersPagingNum, page, 5)
  52. opts.ListOptions.Page = page
  53. opts.ListOptions.PageSize = setting.UI.MembersPagingNum
  54. members, membersIsPublic, err := organization.FindOrgMembers(ctx, opts)
  55. if err != nil {
  56. ctx.ServerError("GetMembers", err)
  57. return
  58. }
  59. ctx.Data["Page"] = pager
  60. ctx.Data["Members"] = members
  61. ctx.Data["MembersIsPublicMember"] = membersIsPublic
  62. ctx.Data["MembersIsUserOrgOwner"] = organization.IsUserOrgOwner(ctx, members, org.ID)
  63. ctx.Data["MembersTwoFaStatus"] = members.GetTwoFaStatus(ctx)
  64. ctx.HTML(http.StatusOK, tplMembers)
  65. }
  66. // MembersAction response for operation to a member of organization
  67. func MembersAction(ctx *context.Context) {
  68. uid := ctx.FormInt64("uid")
  69. if uid == 0 {
  70. ctx.Redirect(ctx.Org.OrgLink + "/members")
  71. return
  72. }
  73. org := ctx.Org.Organization
  74. var err error
  75. switch ctx.Params(":action") {
  76. case "private":
  77. if ctx.Doer.ID != uid && !ctx.Org.IsOwner {
  78. ctx.Error(http.StatusNotFound)
  79. return
  80. }
  81. err = organization.ChangeOrgUserStatus(ctx, org.ID, uid, false)
  82. case "public":
  83. if ctx.Doer.ID != uid && !ctx.Org.IsOwner {
  84. ctx.Error(http.StatusNotFound)
  85. return
  86. }
  87. err = organization.ChangeOrgUserStatus(ctx, org.ID, uid, true)
  88. case "remove":
  89. if !ctx.Org.IsOwner {
  90. ctx.Error(http.StatusNotFound)
  91. return
  92. }
  93. err = models.RemoveOrgUser(ctx, org.ID, uid)
  94. if organization.IsErrLastOrgOwner(err) {
  95. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  96. ctx.JSONRedirect(ctx.Org.OrgLink + "/members")
  97. return
  98. }
  99. case "leave":
  100. err = models.RemoveOrgUser(ctx, org.ID, ctx.Doer.ID)
  101. if err == nil {
  102. ctx.Flash.Success(ctx.Tr("form.organization_leave_success", org.DisplayName()))
  103. ctx.JSON(http.StatusOK, map[string]any{
  104. "redirect": "", // keep the user stay on current page, in case they want to do other operations.
  105. })
  106. } else if organization.IsErrLastOrgOwner(err) {
  107. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  108. ctx.JSONRedirect(ctx.Org.OrgLink + "/members")
  109. } else {
  110. log.Error("RemoveOrgUser(%d,%d): %v", org.ID, ctx.Doer.ID, err)
  111. }
  112. return
  113. }
  114. if err != nil {
  115. log.Error("Action(%s): %v", ctx.Params(":action"), err)
  116. ctx.JSON(http.StatusOK, map[string]any{
  117. "ok": false,
  118. "err": err.Error(),
  119. })
  120. return
  121. }
  122. redirect := ctx.Org.OrgLink + "/members"
  123. if ctx.Params(":action") == "leave" {
  124. redirect = setting.AppSubURL + "/"
  125. }
  126. ctx.JSONRedirect(redirect)
  127. }