3
0
mirror of https://github.com/ergochat/ergo.git synced 2025-02-10 19:40:50 +01:00
ergo/irc/nickname.go

169 lines
6.6 KiB
Go
Raw Normal View History

// Copyright (c) 2012-2014 Jeremy Latt
2017-03-27 22:15:02 +10:00
// Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
// released under the MIT license
package irc
import (
"crypto/rand"
2016-11-16 12:02:22 +10:00
"fmt"
"strings"
2017-11-22 04:41:11 -05:00
"github.com/goshuirc/irc-go/ircfmt"
2019-05-06 23:17:57 -04:00
"github.com/oragono/oragono/irc/history"
2017-11-22 04:41:11 -05:00
"github.com/oragono/oragono/irc/sno"
2019-05-06 23:17:57 -04:00
"github.com/oragono/oragono/irc/utils"
)
var (
restrictedNicknames = []string{
"=scene=", // used for rp commands
2020-05-28 13:16:17 -04:00
"Global", // global announcements on some networks
// common services not implemented by us:
"MemoServ", "BotServ", "OperServ",
}
2020-08-22 22:43:21 -04:00
restrictedCasefoldedNicks = make(utils.StringSet)
restrictedSkeletons = make(utils.StringSet)
)
2020-04-23 01:38:12 -04:00
func performNickChange(server *Server, client *Client, target *Client, session *Session, nickname string, rb *ResponseBuffer) error {
2020-03-16 07:54:50 -04:00
details := target.Details()
hadNick := details.nick != "*"
origNickMask := details.nickMask
2020-09-23 02:30:34 -04:00
isSanick := client != target
2020-10-06 18:04:29 -04:00
assignedNickname, err, back := client.server.clients.SetNick(target, session, nickname, false)
2018-02-03 22:03:36 +10:00
if err == errNicknameInUse {
2020-09-23 02:30:34 -04:00
if !isSanick {
rb.Add(nil, server.name, ERR_NICKNAMEINUSE, details.nick, utils.SafeErrorParam(nickname), client.t("Nickname is already in use"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "NICKNAME_IN_USE", utils.SafeErrorParam(nickname), client.t("Nickname is already in use"))
}
} else if err == errNicknameReserved {
2020-09-23 02:30:34 -04:00
if !isSanick {
rb.Add(nil, server.name, ERR_NICKNAMEINUSE, details.nick, utils.SafeErrorParam(nickname), client.t("Nickname is reserved by a different account"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "NICKNAME_RESERVED", utils.SafeErrorParam(nickname), client.t("Nickname is reserved by a different account"))
}
} else if err == errNicknameInvalid {
2020-09-23 02:30:34 -04:00
if !isSanick {
rb.Add(nil, server.name, ERR_ERRONEUSNICKNAME, details.nick, utils.SafeErrorParam(nickname), client.t("Erroneous nickname"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "NICKNAME_INVALID", utils.SafeErrorParam(nickname), client.t("Erroneous nickname"))
}
2020-03-16 07:54:50 -04:00
} else if err == errNickAccountMismatch {
// this used to use ERR_NICKNAMEINUSE, but it displayed poorly in some clients;
// ERR_UNKNOWNERROR at least has a better chance of displaying our error text
2020-09-23 02:30:34 -04:00
if !isSanick {
rb.Add(nil, server.name, ERR_UNKNOWNERROR, details.nick, "NICK", client.t("You must use your account name as your nickname"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "UNKNOWN_ERROR", utils.SafeErrorParam(nickname), client.t("This user's nickname and account name need to be equal"))
}
2020-03-16 07:54:50 -04:00
} else if err == errNickMissing {
2020-09-23 02:30:34 -04:00
if !isSanick {
rb.Add(nil, server.name, ERR_NONICKNAMEGIVEN, details.nick, client.t("No nickname given"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "NICKNAME_INVALID", utils.SafeErrorParam(nickname), client.t("No nickname given"))
}
2020-05-17 13:39:37 -04:00
} else if err == errNoop {
2020-09-23 02:30:34 -04:00
if !isSanick {
// no message
} else {
rb.Add(nil, server.name, "NOTE", "SANICK", "NOOP", utils.SafeErrorParam(nickname), client.t("Client already had the desired nickname"))
}
2016-11-16 12:02:22 +10:00
} else if err != nil {
2020-09-23 02:30:34 -04:00
client.server.logger.Error("internal", "couldn't change nick", nickname, err.Error())
if !isSanick {
rb.Add(nil, server.name, ERR_UNKNOWNERROR, details.nick, "NICK", client.t("Could not set or change nickname"))
} else {
rb.Add(nil, server.name, "FAIL", "SANICK", "UNKNOWN_ERROR", utils.SafeErrorParam(nickname), client.t("Could not set or change nickname"))
}
}
if err != nil {
2020-04-23 01:38:12 -04:00
return err
}
2017-11-22 04:41:11 -05:00
2020-01-18 23:47:05 -05:00
message := utils.MakeMessage("")
2019-05-06 23:17:57 -04:00
histItem := history.Item{
Type: history.Nick,
Nick: origNickMask,
AccountName: details.accountName,
Message: message,
}
histItem.Params[0] = assignedNickname
2019-05-06 23:17:57 -04:00
client.server.logger.Debug("nick", fmt.Sprintf("%s changed nickname to %s [%s]", origNickMask, assignedNickname, client.NickCasefolded()))
2017-11-22 04:41:11 -05:00
if hadNick {
2019-05-12 03:25:02 -04:00
if client == target {
target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("$%s$r changed nickname to %s"), details.nick, assignedNickname))
2019-05-12 03:25:02 -04:00
} else {
target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("Operator %s changed nickname of $%s$r to %s"), client.Nick(), details.nick, assignedNickname))
2019-05-12 03:25:02 -04:00
}
2019-05-06 23:17:57 -04:00
target.server.whoWas.Append(details.WhoWas)
rb.AddFromClient(message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", assignedNickname)
for session := range target.Friends() {
if session != rb.session {
session.sendFromClientInternal(false, message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", assignedNickname)
}
2017-11-22 04:41:11 -05:00
}
}
2020-05-19 14:12:20 -04:00
if back {
dispatchAwayNotify(session.client, false, "")
}
2020-09-21 20:30:42 -04:00
for _, channel := range target.Channels() {
2020-05-12 12:05:40 -04:00
channel.AddHistoryItem(histItem, details.account)
2019-05-06 23:17:57 -04:00
}
2020-08-06 03:16:58 -04:00
newCfnick := target.NickCasefolded()
if newCfnick != details.nickCasefolded {
client.server.monitorManager.AlertAbout(details.nick, details.nickCasefolded, false)
client.server.monitorManager.AlertAbout(assignedNickname, newCfnick, true)
}
2020-04-23 01:38:12 -04:00
return nil
}
func (server *Server) RandomlyRename(client *Client) {
2020-03-16 07:54:50 -04:00
format := server.Config().Accounts.NickReservation.GuestFormat
buf := make([]byte, 8)
rand.Read(buf)
2020-03-16 07:54:50 -04:00
nick := strings.Replace(format, "*", utils.B32Encoder.EncodeToString(buf), -1)
sessions := client.Sessions()
if len(sessions) == 0 {
// this can happen if they are anonymous and BRB (in general, an always-on
// client has title to its nickname and will never be the victim of
// a call to RandomlyRename)
client.destroy(nil)
return
}
// XXX arbitrarily pick the first session to receive error messages;
// all other sessions receive a `NICK` line same as a friend would
rb := NewResponseBuffer(sessions[0])
performNickChange(server, client, client, nil, nick, rb)
2018-12-28 13:45:55 -05:00
rb.Send(false)
// technically performNickChange can fail to change the nick,
// but if they're still delinquent, the timer will get them later
}
2020-03-16 23:37:52 -04:00
// if force-nick-equals-account is set, account name and nickname must be equal,
// so we need to re-NICK automatically on every login event (IDENTIFY,
// VERIFY, and a REGISTER that auto-verifies). if we can't get the nick
// then we log them out (they will be able to reattach with SASL)
func fixupNickEqualsAccount(client *Client, rb *ResponseBuffer, config *Config) (success bool) {
if !config.Accounts.NickReservation.ForceNickEqualsAccount {
return true
}
if !client.registered {
return true
}
2020-08-29 21:42:27 -04:00
err := performNickChange(client.server, client, client, rb.session, client.AccountName(), rb)
if err != nil && err != errNoop {
2020-03-16 23:37:52 -04:00
client.server.accounts.Logout(client)
nsNotice(rb, client.t("A client is already using that account; try logging out and logging back in with SASL"))
return false
}
return true
}