2020-03-03 02:25:36 +08:00
|
|
|
// Copyright 2020 The Gitea Authors.
|
2022-11-28 02:20:29 +08:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-03-03 02:25:36 +08:00
|
|
|
|
|
|
|
package admin
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2021-04-05 23:30:52 +08:00
|
|
|
"net/http"
|
2020-03-03 02:25:36 +08:00
|
|
|
"net/url"
|
|
|
|
|
2021-09-24 19:32:56 +08:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2021-11-11 15:03:30 +08:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2020-03-03 02:25:36 +08:00
|
|
|
"code.gitea.io/gitea/modules/base"
|
|
|
|
"code.gitea.io/gitea/modules/log"
|
2024-03-01 02:52:49 +08:00
|
|
|
"code.gitea.io/gitea/modules/optional"
|
2020-03-03 02:25:36 +08:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2024-02-27 15:12:22 +08:00
|
|
|
"code.gitea.io/gitea/services/context"
|
2024-07-25 18:11:04 +08:00
|
|
|
"code.gitea.io/gitea/services/user"
|
2020-03-03 02:25:36 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
tplEmails base.TplName = "admin/emails/list"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Emails show all emails
|
|
|
|
func Emails(ctx *context.Context) {
|
|
|
|
ctx.Data["Title"] = ctx.Tr("admin.emails")
|
|
|
|
ctx.Data["PageIsAdminEmails"] = true
|
|
|
|
|
2021-11-28 22:11:58 +08:00
|
|
|
opts := &user_model.SearchEmailOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2020-03-03 02:25:36 +08:00
|
|
|
PageSize: setting.UI.Admin.UserPagingNum,
|
2021-07-29 09:42:15 +08:00
|
|
|
Page: ctx.FormInt("page"),
|
2020-03-03 02:25:36 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if opts.Page <= 1 {
|
|
|
|
opts.Page = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
type ActiveEmail struct {
|
2021-11-28 22:11:58 +08:00
|
|
|
user_model.SearchEmailResult
|
2020-03-03 02:25:36 +08:00
|
|
|
CanChange bool
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
2021-11-28 22:11:58 +08:00
|
|
|
baseEmails []*user_model.SearchEmailResult
|
2020-03-03 02:25:36 +08:00
|
|
|
emails []ActiveEmail
|
|
|
|
count int64
|
|
|
|
err error
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy user_model.SearchEmailOrderBy
|
2020-03-03 02:25:36 +08:00
|
|
|
)
|
|
|
|
|
2021-08-11 08:31:13 +08:00
|
|
|
ctx.Data["SortType"] = ctx.FormString("sort")
|
|
|
|
switch ctx.FormString("sort") {
|
2020-03-03 02:25:36 +08:00
|
|
|
case "email":
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy = user_model.SearchEmailOrderByEmail
|
2020-03-03 02:25:36 +08:00
|
|
|
case "reverseemail":
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy = user_model.SearchEmailOrderByEmailReverse
|
2020-03-03 02:25:36 +08:00
|
|
|
case "username":
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy = user_model.SearchEmailOrderByName
|
2020-03-03 02:25:36 +08:00
|
|
|
case "reverseusername":
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy = user_model.SearchEmailOrderByNameReverse
|
2020-03-03 02:25:36 +08:00
|
|
|
default:
|
|
|
|
ctx.Data["SortType"] = "email"
|
2021-11-28 22:11:58 +08:00
|
|
|
orderBy = user_model.SearchEmailOrderByEmail
|
2020-03-03 02:25:36 +08:00
|
|
|
}
|
|
|
|
|
2021-07-29 09:42:15 +08:00
|
|
|
opts.Keyword = ctx.FormTrim("q")
|
2020-03-03 02:25:36 +08:00
|
|
|
opts.SortType = orderBy
|
2021-08-11 08:31:13 +08:00
|
|
|
if len(ctx.FormString("is_activated")) != 0 {
|
2024-03-01 02:52:49 +08:00
|
|
|
opts.IsActivated = optional.Some(ctx.FormBool("activated"))
|
2020-03-03 02:25:36 +08:00
|
|
|
}
|
2021-08-11 08:31:13 +08:00
|
|
|
if len(ctx.FormString("is_primary")) != 0 {
|
2024-03-01 02:52:49 +08:00
|
|
|
opts.IsPrimary = optional.Some(ctx.FormBool("primary"))
|
2020-03-03 02:25:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(opts.Keyword) == 0 || isKeywordValid(opts.Keyword) {
|
2023-09-15 01:09:32 +08:00
|
|
|
baseEmails, count, err = user_model.SearchEmails(ctx, opts)
|
2020-03-03 02:25:36 +08:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("SearchEmails", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
emails = make([]ActiveEmail, len(baseEmails))
|
|
|
|
for i := range baseEmails {
|
|
|
|
emails[i].SearchEmailResult = *baseEmails[i]
|
|
|
|
// Don't let the admin deactivate its own primary email address
|
|
|
|
// We already know the user is admin
|
2022-03-22 15:03:22 +08:00
|
|
|
emails[i].CanChange = ctx.Doer.ID != emails[i].UID || !emails[i].IsPrimary
|
2020-03-03 02:25:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx.Data["Keyword"] = opts.Keyword
|
|
|
|
ctx.Data["Total"] = count
|
|
|
|
ctx.Data["Emails"] = emails
|
|
|
|
|
|
|
|
pager := context.NewPagination(int(count), opts.PageSize, opts.Page, 5)
|
|
|
|
pager.SetDefaultParams(ctx)
|
|
|
|
ctx.Data["Page"] = pager
|
|
|
|
|
2021-04-05 23:30:52 +08:00
|
|
|
ctx.HTML(http.StatusOK, tplEmails)
|
2020-03-03 02:25:36 +08:00
|
|
|
}
|
|
|
|
|
2022-01-21 01:46:10 +08:00
|
|
|
var nullByte = []byte{0x00}
|
2020-03-03 02:25:36 +08:00
|
|
|
|
|
|
|
func isKeywordValid(keyword string) bool {
|
|
|
|
return !bytes.Contains([]byte(keyword), nullByte)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ActivateEmail serves a POST request for activating/deactivating a user's email
|
|
|
|
func ActivateEmail(ctx *context.Context) {
|
|
|
|
truefalse := map[string]bool{"1": true, "0": false}
|
|
|
|
|
2021-07-29 09:42:15 +08:00
|
|
|
uid := ctx.FormInt64("uid")
|
2021-08-11 08:31:13 +08:00
|
|
|
email := ctx.FormString("email")
|
|
|
|
primary, okp := truefalse[ctx.FormString("primary")]
|
|
|
|
activate, oka := truefalse[ctx.FormString("activate")]
|
2020-03-03 02:25:36 +08:00
|
|
|
|
|
|
|
if uid == 0 || len(email) == 0 || !okp || !oka {
|
2021-04-05 23:30:52 +08:00
|
|
|
ctx.Error(http.StatusBadRequest)
|
2020-03-03 02:25:36 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("Changing activation for User ID: %d, email: %s, primary: %v to %v", uid, email, primary, activate)
|
|
|
|
|
2023-09-15 01:09:32 +08:00
|
|
|
if err := user_model.ActivateUserEmail(ctx, uid, email, activate); err != nil {
|
2021-07-14 04:59:27 +08:00
|
|
|
log.Error("ActivateUserEmail(%v,%v,%v): %v", uid, email, activate, err)
|
2021-11-11 15:03:30 +08:00
|
|
|
if user_model.IsErrEmailAlreadyUsed(err) {
|
2020-03-03 02:25:36 +08:00
|
|
|
ctx.Flash.Error(ctx.Tr("admin.emails.duplicate_active"))
|
|
|
|
} else {
|
|
|
|
ctx.Flash.Error(ctx.Tr("admin.emails.not_updated", err))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Info("Activation for User ID: %d, email: %s, primary: %v changed to %v", uid, email, primary, activate)
|
|
|
|
ctx.Flash.Info(ctx.Tr("admin.emails.updated"))
|
|
|
|
}
|
|
|
|
|
2024-10-10 12:56:49 +08:00
|
|
|
redirect, _ := url.Parse(setting.AppSubURL + "/-/admin/emails")
|
2020-03-03 02:25:36 +08:00
|
|
|
q := url.Values{}
|
2021-07-29 09:42:15 +08:00
|
|
|
if val := ctx.FormTrim("q"); len(val) > 0 {
|
2020-03-03 02:25:36 +08:00
|
|
|
q.Set("q", val)
|
|
|
|
}
|
2021-07-29 09:42:15 +08:00
|
|
|
if val := ctx.FormTrim("sort"); len(val) > 0 {
|
2020-03-03 02:25:36 +08:00
|
|
|
q.Set("sort", val)
|
|
|
|
}
|
2021-07-29 09:42:15 +08:00
|
|
|
if val := ctx.FormTrim("is_primary"); len(val) > 0 {
|
2020-03-03 02:25:36 +08:00
|
|
|
q.Set("is_primary", val)
|
|
|
|
}
|
2021-07-29 09:42:15 +08:00
|
|
|
if val := ctx.FormTrim("is_activated"); len(val) > 0 {
|
2020-03-03 02:25:36 +08:00
|
|
|
q.Set("is_activated", val)
|
|
|
|
}
|
|
|
|
redirect.RawQuery = q.Encode()
|
|
|
|
ctx.Redirect(redirect.String())
|
|
|
|
}
|
2024-07-25 18:11:04 +08:00
|
|
|
|
|
|
|
// DeleteEmail serves a POST request for delete a user's email
|
|
|
|
func DeleteEmail(ctx *context.Context) {
|
2024-11-26 23:36:55 +08:00
|
|
|
u, err := user_model.GetUserByID(ctx, ctx.FormInt64("uid"))
|
2024-07-25 18:11:04 +08:00
|
|
|
if err != nil || u == nil {
|
|
|
|
ctx.ServerError("GetUserByID", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
email, err := user_model.GetEmailAddressByID(ctx, u.ID, ctx.FormInt64("id"))
|
|
|
|
if err != nil || email == nil {
|
|
|
|
ctx.ServerError("GetEmailAddressByID", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := user.DeleteEmailAddresses(ctx, u, []string{email.Email}); err != nil {
|
|
|
|
if user_model.IsErrPrimaryEmailCannotDelete(err) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("admin.emails.delete_primary_email_error"))
|
|
|
|
ctx.JSONRedirect("")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.ServerError("DeleteEmailAddresses", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
log.Trace("Email address deleted: %s %s", u.Name, email.Email)
|
|
|
|
|
|
|
|
ctx.Flash.Success(ctx.Tr("admin.emails.deletion_success"))
|
|
|
|
ctx.JSONRedirect("")
|
|
|
|
}
|