123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 |
- // Copyright (c) 2012-2014 Jeremy Latt
- // Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
- // released under the MIT license
-
- package irc
-
- import (
- "crypto/rand"
- "encoding/hex"
- "fmt"
- "strings"
-
- "github.com/goshuirc/irc-go/ircfmt"
- "github.com/oragono/oragono/irc/history"
- "github.com/oragono/oragono/irc/sno"
- "github.com/oragono/oragono/irc/utils"
- )
-
- var (
- // anything added here MUST be casefolded:
- restrictedNicknames = map[string]bool{
- "=scene=": true, // used for rp commands
- "histserv": true, // TODO(slingamn) this should become a real service
- }
- )
-
- // returns whether the change succeeded or failed
- func performNickChange(server *Server, client *Client, target *Client, session *Session, newnick string, rb *ResponseBuffer) bool {
- nickname := strings.TrimSpace(newnick)
- cfnick, err := CasefoldName(nickname)
- currentNick := client.Nick()
-
- if len(nickname) < 1 {
- rb.Add(nil, server.name, ERR_NONICKNAMEGIVEN, currentNick, client.t("No nickname given"))
- return false
- }
-
- if err != nil || len(nickname) > server.Limits().NickLen || restrictedNicknames[cfnick] {
- rb.Add(nil, server.name, ERR_ERRONEUSNICKNAME, currentNick, nickname, client.t("Erroneous nickname"))
- return false
- }
-
- if target.Nick() == nickname {
- return true
- }
-
- hadNick := target.HasNick()
- origNickMask := target.NickMaskString()
- details := target.Details()
- err = client.server.clients.SetNick(target, session, nickname)
- if err == errNicknameInUse {
- rb.Add(nil, server.name, ERR_NICKNAMEINUSE, currentNick, nickname, client.t("Nickname is already in use"))
- return false
- } else if err == errNicknameReserved {
- rb.Add(nil, server.name, ERR_NICKNAMEINUSE, currentNick, nickname, client.t("Nickname is reserved by a different account"))
- return false
- } else if err != nil {
- rb.Add(nil, server.name, ERR_UNKNOWNERROR, currentNick, "NICK", fmt.Sprintf(client.t("Could not set or change nickname: %s"), err.Error()))
- return false
- }
-
- message := utils.MakeSplitMessage("", true)
- histItem := history.Item{
- Type: history.Nick,
- Nick: origNickMask,
- AccountName: details.accountName,
- Message: message,
- }
- histItem.Params[0] = nickname
-
- client.server.logger.Debug("nick", fmt.Sprintf("%s changed nickname to %s [%s]", origNickMask, nickname, cfnick))
- if hadNick {
- if client == target {
- target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("$%s$r changed nickname to %s"), details.nick, nickname))
- } else {
- target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("Operator %s changed nickname of $%s$r to %s"), client.Nick(), details.nick, nickname))
- }
- target.server.whoWas.Append(details.WhoWas)
- rb.AddFromClient(message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", nickname)
- for session := range target.Friends() {
- if session != rb.session {
- session.sendFromClientInternal(false, message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", nickname)
- }
- }
- }
-
- for _, channel := range client.Channels() {
- channel.history.Add(histItem)
- }
-
- target.nickTimer.Touch(rb)
-
- if target.Registered() {
- client.server.monitorManager.AlertAbout(target, true)
- }
- // else: Run() will attempt registration immediately after this
- return true
- }
-
- func (server *Server) RandomlyRename(client *Client) {
- prefix := server.AccountConfig().NickReservation.RenamePrefix
- if prefix == "" {
- prefix = "Guest-"
- }
- buf := make([]byte, 8)
- rand.Read(buf)
- nick := fmt.Sprintf("%s%s", prefix, hex.EncodeToString(buf))
- sessions := client.Sessions()
- if len(sessions) == 0 {
- 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)
- rb.Send(false)
- // technically performNickChange can fail to change the nick,
- // but if they're still delinquent, the timer will get them later
- }
|