   --------- Co-authored-by: Jason Song <i@wolfogre.com>
		
			
				
	
	
		
			183 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2020 The Gitea Authors.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package admin
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 
 | |
| 	"code.gitea.io/gitea/models/db"
 | |
| 	user_model "code.gitea.io/gitea/models/user"
 | |
| 	"code.gitea.io/gitea/modules/base"
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 	"code.gitea.io/gitea/modules/optional"
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/services/context"
 | |
| 	"code.gitea.io/gitea/services/user"
 | |
| )
 | |
| 
 | |
| 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
 | |
| 
 | |
| 	opts := &user_model.SearchEmailOptions{
 | |
| 		ListOptions: db.ListOptions{
 | |
| 			PageSize: setting.UI.Admin.UserPagingNum,
 | |
| 			Page:     ctx.FormInt("page"),
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	if opts.Page <= 1 {
 | |
| 		opts.Page = 1
 | |
| 	}
 | |
| 
 | |
| 	type ActiveEmail struct {
 | |
| 		user_model.SearchEmailResult
 | |
| 		CanChange bool
 | |
| 	}
 | |
| 
 | |
| 	var (
 | |
| 		baseEmails []*user_model.SearchEmailResult
 | |
| 		emails     []ActiveEmail
 | |
| 		count      int64
 | |
| 		err        error
 | |
| 		orderBy    user_model.SearchEmailOrderBy
 | |
| 	)
 | |
| 
 | |
| 	ctx.Data["SortType"] = ctx.FormString("sort")
 | |
| 	switch ctx.FormString("sort") {
 | |
| 	case "email":
 | |
| 		orderBy = user_model.SearchEmailOrderByEmail
 | |
| 	case "reverseemail":
 | |
| 		orderBy = user_model.SearchEmailOrderByEmailReverse
 | |
| 	case "username":
 | |
| 		orderBy = user_model.SearchEmailOrderByName
 | |
| 	case "reverseusername":
 | |
| 		orderBy = user_model.SearchEmailOrderByNameReverse
 | |
| 	default:
 | |
| 		ctx.Data["SortType"] = "email"
 | |
| 		orderBy = user_model.SearchEmailOrderByEmail
 | |
| 	}
 | |
| 
 | |
| 	opts.Keyword = ctx.FormTrim("q")
 | |
| 	opts.SortType = orderBy
 | |
| 	if len(ctx.FormString("is_activated")) != 0 {
 | |
| 		opts.IsActivated = optional.Some(ctx.FormBool("activated"))
 | |
| 	}
 | |
| 	if len(ctx.FormString("is_primary")) != 0 {
 | |
| 		opts.IsPrimary = optional.Some(ctx.FormBool("primary"))
 | |
| 	}
 | |
| 
 | |
| 	if len(opts.Keyword) == 0 || isKeywordValid(opts.Keyword) {
 | |
| 		baseEmails, count, err = user_model.SearchEmails(ctx, opts)
 | |
| 		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
 | |
| 			emails[i].CanChange = ctx.Doer.ID != emails[i].UID || !emails[i].IsPrimary
 | |
| 		}
 | |
| 	}
 | |
| 	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
 | |
| 
 | |
| 	ctx.HTML(http.StatusOK, tplEmails)
 | |
| }
 | |
| 
 | |
| var nullByte = []byte{0x00}
 | |
| 
 | |
| 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}
 | |
| 
 | |
| 	uid := ctx.FormInt64("uid")
 | |
| 	email := ctx.FormString("email")
 | |
| 	primary, okp := truefalse[ctx.FormString("primary")]
 | |
| 	activate, oka := truefalse[ctx.FormString("activate")]
 | |
| 
 | |
| 	if uid == 0 || len(email) == 0 || !okp || !oka {
 | |
| 		ctx.Error(http.StatusBadRequest)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	log.Info("Changing activation for User ID: %d, email: %s, primary: %v to %v", uid, email, primary, activate)
 | |
| 
 | |
| 	if err := user_model.ActivateUserEmail(ctx, uid, email, activate); err != nil {
 | |
| 		log.Error("ActivateUserEmail(%v,%v,%v): %v", uid, email, activate, err)
 | |
| 		if user_model.IsErrEmailAlreadyUsed(err) {
 | |
| 			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"))
 | |
| 	}
 | |
| 
 | |
| 	redirect, _ := url.Parse(setting.AppSubURL + "/admin/emails")
 | |
| 	q := url.Values{}
 | |
| 	if val := ctx.FormTrim("q"); len(val) > 0 {
 | |
| 		q.Set("q", val)
 | |
| 	}
 | |
| 	if val := ctx.FormTrim("sort"); len(val) > 0 {
 | |
| 		q.Set("sort", val)
 | |
| 	}
 | |
| 	if val := ctx.FormTrim("is_primary"); len(val) > 0 {
 | |
| 		q.Set("is_primary", val)
 | |
| 	}
 | |
| 	if val := ctx.FormTrim("is_activated"); len(val) > 0 {
 | |
| 		q.Set("is_activated", val)
 | |
| 	}
 | |
| 	redirect.RawQuery = q.Encode()
 | |
| 	ctx.Redirect(redirect.String())
 | |
| }
 | |
| 
 | |
| // DeleteEmail serves a POST request for delete a user's email
 | |
| func DeleteEmail(ctx *context.Context) {
 | |
| 	u, err := user_model.GetUserByID(ctx, ctx.FormInt64("Uid"))
 | |
| 	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("")
 | |
| }
 |