193 lines
5.8 KiB
Go
193 lines
5.8 KiB
Go
package v1
|
|
|
|
import (
|
|
"database/sql"
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"zxq.co/ripple/rippleapi/common"
|
|
)
|
|
|
|
type setAllowedData struct {
|
|
UserID int `json:"user_id"`
|
|
Allowed int `json:"allowed"`
|
|
}
|
|
|
|
// UserManageSetAllowedPOST allows to set the allowed status of an user.
|
|
func UserManageSetAllowedPOST(md common.MethodData) common.CodeMessager {
|
|
var data setAllowedData
|
|
if err := md.Unmarshal(&data); err != nil {
|
|
return ErrBadJSON
|
|
}
|
|
if data.Allowed < 0 || data.Allowed > 2 {
|
|
return common.SimpleResponse(400, "Allowed status must be between 0 and 2")
|
|
}
|
|
var banDatetime int64
|
|
var privsSet string
|
|
if data.Allowed == 0 {
|
|
banDatetime = time.Now().Unix()
|
|
privsSet = "privileges = (privileges & ~3)"
|
|
} else {
|
|
banDatetime = 0
|
|
privsSet = "privileges = (privileges | 3)"
|
|
}
|
|
_, err := md.DB.Exec("UPDATE users SET "+privsSet+", ban_datetime = ? WHERE id = ?", banDatetime, data.UserID)
|
|
if err != nil {
|
|
md.Err(err)
|
|
return Err500
|
|
}
|
|
rapLog(md, fmt.Sprintf("changed UserID:%d's allowed to %d. This was done using the API's terrible ManageSetAllowed.", data.UserID, data.Allowed))
|
|
go fixPrivileges(data.UserID, md.DB)
|
|
query := `
|
|
SELECT users.id, users.username, register_datetime, privileges,
|
|
latest_activity, users_stats.username_aka,
|
|
users_stats.country
|
|
FROM users
|
|
LEFT JOIN users_stats
|
|
ON users.id=users_stats.id
|
|
WHERE users.id=?
|
|
LIMIT 1`
|
|
return userPutsSingle(md, md.DB.QueryRowx(query, data.UserID))
|
|
}
|
|
|
|
type userEditData struct {
|
|
ID int `json:"id"`
|
|
Username *string `json:"username"`
|
|
UsernameAKA *string `json:"username_aka"`
|
|
//Privileges *uint64 `json:"privileges"`
|
|
Country *string `json:"country"`
|
|
SilenceInfo *silenceInfo `json:"silence_info"`
|
|
ResetUserpage bool `json:"reset_userpage"`
|
|
//ResetAvatar bool `json:"reset_avatar"`
|
|
}
|
|
|
|
// UserEditPOST allows to edit an user's information.
|
|
func UserEditPOST(md common.MethodData) common.CodeMessager {
|
|
var data userEditData
|
|
if err := md.Unmarshal(&data); err != nil {
|
|
fmt.Println(err)
|
|
return ErrBadJSON
|
|
}
|
|
|
|
if data.ID == 0 {
|
|
return common.SimpleResponse(404, "That user could not be found")
|
|
}
|
|
|
|
var prevUser struct {
|
|
Username string
|
|
Privileges uint64
|
|
}
|
|
err := md.DB.Get(&prevUser, "SELECT username, privileges FROM users WHERE id = ? LIMIT 1", data.ID)
|
|
|
|
switch err {
|
|
case nil: // carry on
|
|
case sql.ErrNoRows:
|
|
return common.SimpleResponse(404, "That user could not be found")
|
|
default:
|
|
md.Err(err)
|
|
return Err500
|
|
}
|
|
|
|
const initQuery = "UPDATE users SET\n"
|
|
q := initQuery
|
|
var args []interface{}
|
|
|
|
// totally did not realise I had to update some fields in users_stats as well
|
|
// and just copy pasting the above code by prefixing "stats" to every
|
|
// variable
|
|
const statsInitQuery = "UPDATE users_stats SET\n"
|
|
statsQ := statsInitQuery
|
|
var statsArgs []interface{}
|
|
|
|
if common.UserPrivileges(prevUser.Privileges)&common.AdminPrivilegeManageUsers != 0 &&
|
|
data.ID != md.User.UserID {
|
|
return common.SimpleResponse(403, "Can't edit that user")
|
|
}
|
|
|
|
if data.Username != nil {
|
|
if strings.Contains(*data.Username, " ") && strings.Contains(*data.Username, "_") {
|
|
return common.SimpleResponse(400, "Mixed spaces and underscores")
|
|
}
|
|
if usernameAvailable(md, *data.Username, data.ID) {
|
|
return common.SimpleResponse(409, "User with that username exists")
|
|
}
|
|
jsonData, _ := json.Marshal(struct {
|
|
UserID int `json:"userID"`
|
|
NewUsername string `json:"newUsername"`
|
|
}{data.ID, *data.Username})
|
|
md.R.Publish("peppy:change_username", string(jsonData))
|
|
appendToUserNotes(md, "Username change: "+prevUser.Username+" -> "+*data.Username, data.ID)
|
|
}
|
|
if data.UsernameAKA != nil {
|
|
statsQ += "username_aka = ?,\n"
|
|
statsArgs = append(statsArgs, *data.UsernameAKA)
|
|
}
|
|
/*if data.Privileges != nil {
|
|
q += "privileges = ?,\n"
|
|
args = append(args, *data.Privileges)
|
|
// UserNormal or UserPublic changed
|
|
if *data.Privileges & 3 != 3 && *data.Privileges & 3 != prevUser.Privileges & 3 {
|
|
q += "ban_datetime = ?"
|
|
args = append(args, meme)
|
|
}
|
|
// https://zxq.co/ripple/old-frontend/src/master/inc/Do.php#L355 ?
|
|
// should also check for AdminManagePrivileges
|
|
// should also check out the code for CM restring/banning
|
|
}*/
|
|
if data.Country != nil {
|
|
statsQ += "country = ?,\n"
|
|
statsArgs = append(statsArgs, *data.Country)
|
|
rapLog(md, fmt.Sprintf("has changed %s country to %s", prevUser.Username, *data.Country))
|
|
appendToUserNotes(md, "country changed to "+*data.Country, data.ID)
|
|
}
|
|
if data.SilenceInfo != nil && md.User.UserPrivileges&common.AdminPrivilegeSilenceUsers != 0 {
|
|
q += "silence_end = ?, silence_reason = ?,\n"
|
|
args = append(args, time.Time(data.SilenceInfo.End).Unix(), data.SilenceInfo.Reason)
|
|
}
|
|
if data.ResetUserpage {
|
|
statsQ += "userpage_content = '',\n"
|
|
}
|
|
|
|
if q != initQuery {
|
|
q = q[:len(q)-2] + " WHERE id = ? LIMIT 1"
|
|
args = append(args, data.ID)
|
|
_, err = md.DB.Exec(q, args...)
|
|
if err != nil {
|
|
md.Err(err)
|
|
return Err500
|
|
}
|
|
}
|
|
if statsQ != statsInitQuery {
|
|
statsQ = statsQ[:len(statsQ)-2] + " WHERE id = ? LIMIT 1"
|
|
statsArgs = append(statsArgs, data.ID)
|
|
_, err = md.DB.Exec(statsQ, statsArgs...)
|
|
if err != nil {
|
|
md.Err(err)
|
|
return Err500
|
|
}
|
|
}
|
|
|
|
rapLog(md, fmt.Sprintf("has updated user %s", prevUser.Username))
|
|
|
|
return userPutsSingle(md, md.DB.QueryRowx(userFields+" WHERE users.id = ? LIMIT 1", data.ID))
|
|
}
|
|
|
|
func appendToUserNotes(md common.MethodData, message string, user int) {
|
|
message = "\n[" + time.Now().Format("2006-01-02 15:04:05") + "] API: " + message
|
|
_, err := md.DB.Exec("UPDATE users SET notes = CONCAT(COALESCE(notes, ''), ?) WHERE id = ?",
|
|
message, user)
|
|
if err != nil {
|
|
md.Err(err)
|
|
}
|
|
}
|
|
|
|
func usernameAvailable(md common.MethodData, u string, userID int) (r bool) {
|
|
err := md.DB.QueryRow("SELECT EXISTS(SELECT 1 FROM users WHERE username_safe = ? AND id != ?)", common.SafeUsername(u), userID).Scan(&r)
|
|
if err != nil && err != sql.ErrNoRows {
|
|
md.Err(err)
|
|
}
|
|
return
|
|
}
|