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.

emails.go 4.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // Copyright 2020 The Gitea Authors.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package admin
  5. import (
  6. "bytes"
  7. "net/url"
  8. "code.gitea.io/gitea/models"
  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. "code.gitea.io/gitea/modules/util"
  14. "github.com/unknwon/com"
  15. )
  16. const (
  17. tplEmails base.TplName = "admin/emails/list"
  18. )
  19. // Emails show all emails
  20. func Emails(ctx *context.Context) {
  21. ctx.Data["Title"] = ctx.Tr("admin.emails")
  22. ctx.Data["PageIsAdmin"] = true
  23. ctx.Data["PageIsAdminEmails"] = true
  24. opts := &models.SearchEmailOptions{
  25. PageSize: setting.UI.Admin.UserPagingNum,
  26. Page: ctx.QueryInt("page"),
  27. }
  28. if opts.Page <= 1 {
  29. opts.Page = 1
  30. }
  31. type ActiveEmail struct {
  32. models.SearchEmailResult
  33. CanChange bool
  34. }
  35. var (
  36. baseEmails []*models.SearchEmailResult
  37. emails []ActiveEmail
  38. count int64
  39. err error
  40. orderBy models.SearchEmailOrderBy
  41. )
  42. ctx.Data["SortType"] = ctx.Query("sort")
  43. switch ctx.Query("sort") {
  44. case "email":
  45. orderBy = models.SearchEmailOrderByEmail
  46. case "reverseemail":
  47. orderBy = models.SearchEmailOrderByEmailReverse
  48. case "username":
  49. orderBy = models.SearchEmailOrderByName
  50. case "reverseusername":
  51. orderBy = models.SearchEmailOrderByNameReverse
  52. default:
  53. ctx.Data["SortType"] = "email"
  54. orderBy = models.SearchEmailOrderByEmail
  55. }
  56. opts.Keyword = ctx.QueryTrim("q")
  57. opts.SortType = orderBy
  58. if len(ctx.Query("is_activated")) != 0 {
  59. opts.IsActivated = util.OptionalBoolOf(ctx.QueryBool("activated"))
  60. }
  61. if len(ctx.Query("is_primary")) != 0 {
  62. opts.IsPrimary = util.OptionalBoolOf(ctx.QueryBool("primary"))
  63. }
  64. if len(opts.Keyword) == 0 || isKeywordValid(opts.Keyword) {
  65. baseEmails, count, err = models.SearchEmails(opts)
  66. if err != nil {
  67. ctx.ServerError("SearchEmails", err)
  68. return
  69. }
  70. emails = make([]ActiveEmail, len(baseEmails))
  71. for i := range baseEmails {
  72. emails[i].SearchEmailResult = *baseEmails[i]
  73. // Don't let the admin deactivate its own primary email address
  74. // We already know the user is admin
  75. emails[i].CanChange = ctx.User.ID != emails[i].UID || !emails[i].IsPrimary
  76. }
  77. }
  78. ctx.Data["Keyword"] = opts.Keyword
  79. ctx.Data["Total"] = count
  80. ctx.Data["Emails"] = emails
  81. pager := context.NewPagination(int(count), opts.PageSize, opts.Page, 5)
  82. pager.SetDefaultParams(ctx)
  83. ctx.Data["Page"] = pager
  84. ctx.HTML(200, tplEmails)
  85. }
  86. var (
  87. nullByte = []byte{0x00}
  88. )
  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 := com.StrTo(ctx.Query("uid")).MustInt64()
  96. email := ctx.Query("email")
  97. primary, okp := truefalse[ctx.Query("primary")]
  98. activate, oka := truefalse[ctx.Query("activate")]
  99. if uid == 0 || len(email) == 0 || !okp || !oka {
  100. ctx.Error(400)
  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 := models.ActivateUserEmail(uid, email, primary, activate); err != nil {
  105. log.Error("ActivateUserEmail(%v,%v,%v,%v): %v", uid, email, primary, activate, err)
  106. if models.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.QueryTrim("q"); len(val) > 0 {
  118. q.Set("q", val)
  119. }
  120. if val := ctx.QueryTrim("sort"); len(val) > 0 {
  121. q.Set("sort", val)
  122. }
  123. if val := ctx.QueryTrim("is_primary"); len(val) > 0 {
  124. q.Set("is_primary", val)
  125. }
  126. if val := ctx.QueryTrim("is_activated"); len(val) > 0 {
  127. q.Set("is_activated", val)
  128. }
  129. redirect.RawQuery = q.Encode()
  130. ctx.Redirect(redirect.String())
  131. }