gitea源码

emails.go 5.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. // Copyright 2020 The Gitea Authors.
  2. // SPDX-License-Identifier: MIT
  3. package admin
  4. import (
  5. "bytes"
  6. "net/http"
  7. "net/url"
  8. "code.gitea.io/gitea/models/db"
  9. user_model "code.gitea.io/gitea/models/user"
  10. "code.gitea.io/gitea/modules/log"
  11. "code.gitea.io/gitea/modules/optional"
  12. "code.gitea.io/gitea/modules/setting"
  13. "code.gitea.io/gitea/modules/templates"
  14. "code.gitea.io/gitea/services/context"
  15. "code.gitea.io/gitea/services/user"
  16. )
  17. const (
  18. tplEmails templates.TplName = "admin/emails/list"
  19. )
  20. // Emails show all emails
  21. func Emails(ctx *context.Context) {
  22. ctx.Data["Title"] = ctx.Tr("admin.emails")
  23. ctx.Data["PageIsAdminEmails"] = true
  24. opts := &user_model.SearchEmailOptions{
  25. ListOptions: db.ListOptions{
  26. PageSize: setting.UI.Admin.UserPagingNum,
  27. Page: ctx.FormInt("page"),
  28. },
  29. }
  30. if opts.Page <= 1 {
  31. opts.Page = 1
  32. }
  33. type ActiveEmail struct {
  34. user_model.SearchEmailResult
  35. CanChange bool
  36. }
  37. var (
  38. baseEmails []*user_model.SearchEmailResult
  39. emails []ActiveEmail
  40. count int64
  41. err error
  42. orderBy user_model.SearchEmailOrderBy
  43. )
  44. ctx.Data["SortType"] = ctx.FormString("sort")
  45. switch ctx.FormString("sort") {
  46. case "email":
  47. orderBy = user_model.SearchEmailOrderByEmail
  48. case "reverseemail":
  49. orderBy = user_model.SearchEmailOrderByEmailReverse
  50. case "username":
  51. orderBy = user_model.SearchEmailOrderByName
  52. case "reverseusername":
  53. orderBy = user_model.SearchEmailOrderByNameReverse
  54. default:
  55. ctx.Data["SortType"] = "email"
  56. orderBy = user_model.SearchEmailOrderByEmail
  57. }
  58. opts.Keyword = ctx.FormTrim("q")
  59. opts.SortType = orderBy
  60. if len(ctx.FormString("is_activated")) != 0 {
  61. opts.IsActivated = optional.Some(ctx.FormBool("activated"))
  62. }
  63. if len(ctx.FormString("is_primary")) != 0 {
  64. opts.IsPrimary = optional.Some(ctx.FormBool("primary"))
  65. }
  66. if len(opts.Keyword) == 0 || isKeywordValid(opts.Keyword) {
  67. baseEmails, count, err = user_model.SearchEmails(ctx, opts)
  68. if err != nil {
  69. ctx.ServerError("SearchEmails", err)
  70. return
  71. }
  72. emails = make([]ActiveEmail, len(baseEmails))
  73. for i := range baseEmails {
  74. emails[i].SearchEmailResult = *baseEmails[i]
  75. // Don't let the admin deactivate its own primary email address
  76. // We already know the user is admin
  77. emails[i].CanChange = ctx.Doer.ID != emails[i].UID || !emails[i].IsPrimary
  78. }
  79. }
  80. ctx.Data["Keyword"] = opts.Keyword
  81. ctx.Data["Total"] = count
  82. ctx.Data["Emails"] = emails
  83. pager := context.NewPagination(int(count), opts.PageSize, opts.Page, 5)
  84. pager.AddParamFromRequest(ctx.Req)
  85. ctx.Data["Page"] = pager
  86. ctx.HTML(http.StatusOK, tplEmails)
  87. }
  88. var nullByte = []byte{0x00}
  89. func isKeywordValid(keyword string) bool {
  90. return !bytes.Contains([]byte(keyword), nullByte)
  91. }
  92. // ActivateEmail serves a POST request for activating/deactivating a user's email
  93. func ActivateEmail(ctx *context.Context) {
  94. truefalse := map[string]bool{"1": true, "0": false}
  95. uid := ctx.FormInt64("uid")
  96. email := ctx.FormString("email")
  97. primary, okp := truefalse[ctx.FormString("primary")]
  98. activate, oka := truefalse[ctx.FormString("activate")]
  99. if uid == 0 || len(email) == 0 || !okp || !oka {
  100. ctx.HTTPError(http.StatusBadRequest)
  101. return
  102. }
  103. log.Info("Changing activation for User ID: %d, email: %s, primary: %v to %v", uid, email, primary, activate)
  104. if err := user_model.ActivateUserEmail(ctx, uid, email, activate); err != nil {
  105. log.Error("ActivateUserEmail(%v,%v,%v): %v", uid, email, activate, err)
  106. if user_model.IsErrEmailAlreadyUsed(err) {
  107. ctx.Flash.Error(ctx.Tr("admin.emails.duplicate_active"))
  108. } else {
  109. ctx.Flash.Error(ctx.Tr("admin.emails.not_updated", err))
  110. }
  111. } else {
  112. log.Info("Activation for User ID: %d, email: %s, primary: %v changed to %v", uid, email, primary, activate)
  113. ctx.Flash.Info(ctx.Tr("admin.emails.updated"))
  114. }
  115. redirect, _ := url.Parse(setting.AppSubURL + "/-/admin/emails")
  116. q := url.Values{}
  117. if val := ctx.FormTrim("q"); len(val) > 0 {
  118. q.Set("q", val)
  119. }
  120. if val := ctx.FormTrim("sort"); len(val) > 0 {
  121. q.Set("sort", val)
  122. }
  123. if val := ctx.FormTrim("is_primary"); len(val) > 0 {
  124. q.Set("is_primary", val)
  125. }
  126. if val := ctx.FormTrim("is_activated"); len(val) > 0 {
  127. q.Set("is_activated", val)
  128. }
  129. redirect.RawQuery = q.Encode()
  130. ctx.Redirect(redirect.String())
  131. }
  132. // DeleteEmail serves a POST request for delete a user's email
  133. func DeleteEmail(ctx *context.Context) {
  134. u, err := user_model.GetUserByID(ctx, ctx.FormInt64("uid"))
  135. if err != nil || u == nil {
  136. ctx.ServerError("GetUserByID", err)
  137. return
  138. }
  139. email, err := user_model.GetEmailAddressByID(ctx, u.ID, ctx.FormInt64("id"))
  140. if err != nil || email == nil {
  141. ctx.ServerError("GetEmailAddressByID", err)
  142. return
  143. }
  144. if err := user.DeleteEmailAddresses(ctx, u, []string{email.Email}); err != nil {
  145. if user_model.IsErrPrimaryEmailCannotDelete(err) {
  146. ctx.Flash.Error(ctx.Tr("admin.emails.delete_primary_email_error"))
  147. ctx.JSONRedirect("")
  148. return
  149. }
  150. ctx.ServerError("DeleteEmailAddresses", err)
  151. return
  152. }
  153. log.Trace("Email address deleted: %s %s", u.Name, email.Email)
  154. ctx.Flash.Success(ctx.Tr("admin.emails.deletion_success"))
  155. ctx.JSONRedirect("")
  156. }