You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

nickname.go 2.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. // Copyright (c) 2012-2014 Jeremy Latt
  2. // Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
  3. // released under the MIT license
  4. package irc
  5. import (
  6. "crypto/rand"
  7. "encoding/hex"
  8. "fmt"
  9. "strings"
  10. "github.com/goshuirc/irc-go/ircfmt"
  11. "github.com/oragono/oragono/irc/sno"
  12. )
  13. var (
  14. // anything added here MUST be casefolded:
  15. restrictedNicknames = map[string]bool{
  16. "=scene=": true, // used for rp commands
  17. "histserv": true, // TODO(slingamn) this should become a real service
  18. }
  19. )
  20. // returns whether the change succeeded or failed
  21. func performNickChange(server *Server, client *Client, target *Client, newnick string, rb *ResponseBuffer) bool {
  22. nickname := strings.TrimSpace(newnick)
  23. cfnick, err := CasefoldName(nickname)
  24. if len(nickname) < 1 {
  25. rb.Add(nil, server.name, ERR_NONICKNAMEGIVEN, client.nick, client.t("No nickname given"))
  26. return false
  27. }
  28. if err != nil || len(nickname) > server.Limits().NickLen || restrictedNicknames[cfnick] {
  29. rb.Add(nil, server.name, ERR_ERRONEUSNICKNAME, client.nick, nickname, client.t("Erroneous nickname"))
  30. return false
  31. }
  32. if target.Nick() == nickname {
  33. return true
  34. }
  35. hadNick := target.HasNick()
  36. origNickMask := target.NickMaskString()
  37. whowas := client.WhoWas()
  38. err = client.server.clients.SetNick(target, nickname)
  39. if err == errNicknameInUse {
  40. rb.Add(nil, server.name, ERR_NICKNAMEINUSE, client.nick, nickname, client.t("Nickname is already in use"))
  41. return false
  42. } else if err == errNicknameReserved {
  43. rb.Add(nil, server.name, ERR_NICKNAMEINUSE, client.nick, nickname, client.t("Nickname is reserved by a different account"))
  44. return false
  45. } else if err != nil {
  46. rb.Add(nil, server.name, ERR_UNKNOWNERROR, client.nick, "NICK", fmt.Sprintf(client.t("Could not set or change nickname: %s"), err.Error()))
  47. return false
  48. }
  49. client.nickTimer.Touch()
  50. client.server.logger.Debug("nick", fmt.Sprintf("%s changed nickname to %s [%s]", origNickMask, nickname, cfnick))
  51. if hadNick {
  52. target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("$%s$r changed nickname to %s"), whowas.nick, nickname))
  53. target.server.whoWas.Append(whowas)
  54. rb.Add(nil, origNickMask, "NICK", nickname)
  55. for friend := range target.Friends() {
  56. if friend != client {
  57. friend.Send(nil, origNickMask, "NICK", nickname)
  58. }
  59. }
  60. }
  61. if target.Registered() {
  62. client.server.monitorManager.AlertAbout(target, true)
  63. }
  64. // else: Run() will attempt registration immediately after this
  65. return true
  66. }
  67. func (server *Server) RandomlyRename(client *Client) {
  68. prefix := server.AccountConfig().NickReservation.RenamePrefix
  69. if prefix == "" {
  70. prefix = "Guest-"
  71. }
  72. buf := make([]byte, 8)
  73. rand.Read(buf)
  74. nick := fmt.Sprintf("%s%s", prefix, hex.EncodeToString(buf))
  75. rb := NewResponseBuffer(client)
  76. performNickChange(server, client, client, nick, rb)
  77. rb.Send(false)
  78. // technically performNickChange can fail to change the nick,
  79. // but if they're still delinquent, the timer will get them later
  80. }