2018-05-17 06:05:00 +02:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// Copyright 2018 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package setting
import (
2019-08-29 14:05:42 +00:00
"errors"
2018-05-17 06:05:00 +02:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/auth"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/context"
"code.gitea.io/gitea/modules/log"
2019-10-14 22:24:26 +07:00
"code.gitea.io/gitea/modules/password"
2018-05-17 06:05:00 +02:00
"code.gitea.io/gitea/modules/setting"
2019-08-15 22:46:21 +08:00
"code.gitea.io/gitea/modules/timeutil"
2019-09-24 13:02:49 +08:00
"code.gitea.io/gitea/services/mailer"
2018-05-17 06:05:00 +02:00
)
const (
tplSettingsAccount base . TplName = "user/settings/account"
)
// Account renders change user's password, user's email and user suicide page
func Account ( ctx * context . Context ) {
ctx . Data [ "Title" ] = ctx . Tr ( "settings" )
ctx . Data [ "PageIsSettingsAccount" ] = true
ctx . Data [ "Email" ] = ctx . User . Email
2019-08-29 14:05:42 +00:00
ctx . Data [ "EmailNotificationsPreference" ] = ctx . User . EmailNotifications ( )
2018-05-17 06:05:00 +02:00
2018-06-18 20:24:45 +02:00
loadAccountData ( ctx )
2018-05-17 06:05:00 +02:00
ctx . HTML ( 200 , tplSettingsAccount )
}
// AccountPost response for change user's password
func AccountPost ( ctx * context . Context , form auth . ChangePasswordForm ) {
ctx . Data [ "Title" ] = ctx . Tr ( "settings" )
ctx . Data [ "PageIsSettingsAccount" ] = true
if ctx . HasError ( ) {
2018-06-18 20:24:45 +02:00
loadAccountData ( ctx )
2018-05-17 06:05:00 +02:00
ctx . HTML ( 200 , tplSettingsAccount )
return
}
if len ( form . Password ) < setting . MinPasswordLength {
ctx . Flash . Error ( ctx . Tr ( "auth.password_too_short" , setting . MinPasswordLength ) )
} else if ctx . User . IsPasswordSet ( ) && ! ctx . User . ValidatePassword ( form . OldPassword ) {
ctx . Flash . Error ( ctx . Tr ( "settings.password_incorrect" ) )
} else if form . Password != form . Retype {
ctx . Flash . Error ( ctx . Tr ( "form.password_not_match" ) )
2019-10-14 22:24:26 +07:00
} else if ! password . IsComplexEnough ( form . Password ) {
ctx . Flash . Error ( ctx . Tr ( "settings.password_complexity" ) )
2018-05-17 06:05:00 +02:00
} else {
var err error
if ctx . User . Salt , err = models . GetUserSalt ( ) ; err != nil {
ctx . ServerError ( "UpdateUser" , err )
return
}
ctx . User . HashPassword ( form . Password )
if err := models . UpdateUserCols ( ctx . User , "salt" , "passwd" ) ; err != nil {
ctx . ServerError ( "UpdateUser" , err )
return
}
log . Trace ( "User password updated: %s" , ctx . User . Name )
ctx . Flash . Success ( ctx . Tr ( "settings.change_password_success" ) )
}
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
}
// EmailPost response for change user's email
func EmailPost ( ctx * context . Context , form auth . AddEmailForm ) {
ctx . Data [ "Title" ] = ctx . Tr ( "settings" )
ctx . Data [ "PageIsSettingsAccount" ] = true
// Make emailaddress primary.
if ctx . Query ( "_method" ) == "PRIMARY" {
if err := models . MakeEmailPrimary ( & models . EmailAddress { ID : ctx . QueryInt64 ( "id" ) } ) ; err != nil {
ctx . ServerError ( "MakeEmailPrimary" , err )
return
}
log . Trace ( "Email made primary: %s" , ctx . User . Name )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
return
}
2019-08-29 14:05:42 +00:00
// Set Email Notification Preference
if ctx . Query ( "_method" ) == "NOTIFICATION" {
preference := ctx . Query ( "preference" )
if ! ( preference == models . EmailNotificationsEnabled ||
preference == models . EmailNotificationsOnMention ||
preference == models . EmailNotificationsDisabled ) {
log . Error ( "Email notifications preference change returned unrecognized option %s: %s" , preference , ctx . User . Name )
ctx . ServerError ( "SetEmailPreference" , errors . New ( "option unrecognized" ) )
return
}
if err := ctx . User . SetEmailNotifications ( preference ) ; err != nil {
log . Error ( "Set Email Notifications failed: %v" , err )
ctx . ServerError ( "SetEmailNotifications" , err )
return
}
log . Trace ( "Email notifications preference made %s: %s" , preference , ctx . User . Name )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
return
}
2018-05-17 06:05:00 +02:00
if ctx . HasError ( ) {
2018-06-18 20:24:45 +02:00
loadAccountData ( ctx )
2018-05-17 06:05:00 +02:00
ctx . HTML ( 200 , tplSettingsAccount )
return
}
email := & models . EmailAddress {
UID : ctx . User . ID ,
Email : form . Email ,
IsActivated : ! setting . Service . RegisterEmailConfirm ,
}
if err := models . AddEmailAddress ( email ) ; err != nil {
if models . IsErrEmailAlreadyUsed ( err ) {
2018-06-18 20:24:45 +02:00
loadAccountData ( ctx )
2018-05-17 06:05:00 +02:00
ctx . RenderWithErr ( ctx . Tr ( "form.email_been_used" ) , tplSettingsAccount , & form )
return
}
ctx . ServerError ( "AddEmailAddress" , err )
return
}
// Send confirmation email
if setting . Service . RegisterEmailConfirm {
2019-09-24 13:02:49 +08:00
mailer . SendActivateEmailMail ( ctx . Locale , ctx . User , email )
2018-05-17 06:05:00 +02:00
if err := ctx . Cache . Put ( "MailResendLimit_" + ctx . User . LowerName , ctx . User . LowerName , 180 ) ; err != nil {
2019-04-02 08:48:31 +01:00
log . Error ( "Set cache(MailResendLimit) fail: %v" , err )
2018-05-17 06:05:00 +02:00
}
2019-08-15 22:46:21 +08:00
ctx . Flash . Info ( ctx . Tr ( "settings.add_email_confirmation_sent" , email . Email , timeutil . MinutesToFriendly ( setting . Service . ActiveCodeLives , ctx . Locale . Language ( ) ) ) )
2018-05-17 06:05:00 +02:00
} else {
ctx . Flash . Success ( ctx . Tr ( "settings.add_email_success" ) )
}
log . Trace ( "Email address added: %s" , email . Email )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
}
// DeleteEmail response for delete user's email
func DeleteEmail ( ctx * context . Context ) {
if err := models . DeleteEmailAddress ( & models . EmailAddress { ID : ctx . QueryInt64 ( "id" ) , UID : ctx . User . ID } ) ; err != nil {
ctx . ServerError ( "DeleteEmail" , err )
return
}
log . Trace ( "Email address deleted: %s" , ctx . User . Name )
ctx . Flash . Success ( ctx . Tr ( "settings.email_deletion_success" ) )
ctx . JSON ( 200 , map [ string ] interface { } {
"redirect" : setting . AppSubURL + "/user/settings/account" ,
} )
}
// DeleteAccount render user suicide page and response for delete user himself
func DeleteAccount ( ctx * context . Context ) {
ctx . Data [ "Title" ] = ctx . Tr ( "settings" )
ctx . Data [ "PageIsSettingsAccount" ] = true
if _ , err := models . UserSignIn ( ctx . User . Name , ctx . Query ( "password" ) ) ; err != nil {
if models . IsErrUserNotExist ( err ) {
2018-06-18 20:24:45 +02:00
loadAccountData ( ctx )
2018-05-17 06:05:00 +02:00
ctx . RenderWithErr ( ctx . Tr ( "form.enterred_invalid_password" ) , tplSettingsAccount , nil )
} else {
ctx . ServerError ( "UserSignIn" , err )
}
return
}
if err := models . DeleteUser ( ctx . User ) ; err != nil {
switch {
case models . IsErrUserOwnRepos ( err ) :
ctx . Flash . Error ( ctx . Tr ( "form.still_own_repo" ) )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
case models . IsErrUserHasOrgs ( err ) :
ctx . Flash . Error ( ctx . Tr ( "form.still_has_org" ) )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
default :
ctx . ServerError ( "DeleteUser" , err )
}
} else {
log . Trace ( "Account deleted: %s" , ctx . User . Name )
ctx . Redirect ( setting . AppSubURL + "/" )
}
}
2018-06-18 20:24:45 +02:00
2019-01-09 18:22:57 +01:00
// UpdateUIThemePost is used to update users' specific theme
func UpdateUIThemePost ( ctx * context . Context , form auth . UpdateThemeForm ) {
ctx . Data [ "Title" ] = ctx . Tr ( "settings" )
ctx . Data [ "PageIsSettingsAccount" ] = true
if ctx . HasError ( ) {
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
return
}
if ! form . IsThemeExists ( ) {
ctx . Flash . Error ( ctx . Tr ( "settings.theme_update_error" ) )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
return
}
if err := ctx . User . UpdateTheme ( form . Theme ) ; err != nil {
ctx . Flash . Error ( ctx . Tr ( "settings.theme_update_error" ) )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
return
}
log . Trace ( "Update user theme: %s" , ctx . User . Name )
ctx . Flash . Success ( ctx . Tr ( "settings.theme_update_success" ) )
ctx . Redirect ( setting . AppSubURL + "/user/settings/account" )
}
2018-06-18 20:24:45 +02:00
func loadAccountData ( ctx * context . Context ) {
emails , err := models . GetEmailAddresses ( ctx . User . ID )
if err != nil {
ctx . ServerError ( "GetEmailAddresses" , err )
return
}
ctx . Data [ "Emails" ] = emails
}