Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

nickname.go 5.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  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. "fmt"
  8. "strings"
  9. "github.com/goshuirc/irc-go/ircfmt"
  10. "github.com/oragono/oragono/irc/history"
  11. "github.com/oragono/oragono/irc/sno"
  12. "github.com/oragono/oragono/irc/utils"
  13. )
  14. var (
  15. restrictedNicknames = []string{
  16. "=scene=", // used for rp commands
  17. }
  18. restrictedCasefoldedNicks = make(map[string]bool)
  19. restrictedSkeletons = make(map[string]bool)
  20. )
  21. func performNickChange(server *Server, client *Client, target *Client, session *Session, nickname string, rb *ResponseBuffer) error {
  22. currentNick := client.Nick()
  23. details := target.Details()
  24. if details.nick == nickname {
  25. return nil
  26. }
  27. hadNick := details.nick != "*"
  28. origNickMask := details.nickMask
  29. assignedNickname, err, back := client.server.clients.SetNick(target, session, nickname)
  30. if err == errNicknameInUse {
  31. rb.Add(nil, server.name, ERR_NICKNAMEINUSE, currentNick, utils.SafeErrorParam(nickname), client.t("Nickname is already in use"))
  32. } else if err == errNicknameReserved {
  33. rb.Add(nil, server.name, ERR_NICKNAMEINUSE, currentNick, utils.SafeErrorParam(nickname), client.t("Nickname is reserved by a different account"))
  34. } else if err == errNicknameInvalid {
  35. rb.Add(nil, server.name, ERR_ERRONEUSNICKNAME, currentNick, utils.SafeErrorParam(nickname), client.t("Erroneous nickname"))
  36. } else if err == errNickAccountMismatch {
  37. // this used to use ERR_NICKNAMEINUSE, but it displayed poorly in some clients;
  38. // ERR_UNKNOWNERROR at least has a better chance of displaying our error text
  39. rb.Add(nil, server.name, ERR_UNKNOWNERROR, currentNick, "NICK", client.t("You must use your account name as your nickname"))
  40. } else if err == errNickMissing {
  41. rb.Add(nil, server.name, ERR_NONICKNAMEGIVEN, currentNick, client.t("No nickname given"))
  42. } else if err == errNoop {
  43. // no message
  44. } else if err != nil {
  45. rb.Add(nil, server.name, ERR_UNKNOWNERROR, currentNick, "NICK", fmt.Sprintf(client.t("Could not set or change nickname: %s"), err.Error()))
  46. }
  47. if err != nil {
  48. return err
  49. }
  50. message := utils.MakeMessage("")
  51. histItem := history.Item{
  52. Type: history.Nick,
  53. Nick: origNickMask,
  54. AccountName: details.accountName,
  55. Message: message,
  56. }
  57. histItem.Params[0] = assignedNickname
  58. client.server.logger.Debug("nick", fmt.Sprintf("%s changed nickname to %s [%s]", origNickMask, assignedNickname, client.NickCasefolded()))
  59. if hadNick {
  60. if client == target {
  61. target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("$%s$r changed nickname to %s"), details.nick, assignedNickname))
  62. } else {
  63. target.server.snomasks.Send(sno.LocalNicks, fmt.Sprintf(ircfmt.Unescape("Operator %s changed nickname of $%s$r to %s"), client.Nick(), details.nick, assignedNickname))
  64. }
  65. target.server.whoWas.Append(details.WhoWas)
  66. rb.AddFromClient(message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", assignedNickname)
  67. for session := range target.Friends() {
  68. if session != rb.session {
  69. session.sendFromClientInternal(false, message.Time, message.Msgid, origNickMask, details.accountName, nil, "NICK", assignedNickname)
  70. }
  71. }
  72. }
  73. if back {
  74. dispatchAwayNotify(session.client, false, "")
  75. }
  76. for _, channel := range client.Channels() {
  77. channel.AddHistoryItem(histItem, details.account)
  78. }
  79. if target.Registered() {
  80. client.server.monitorManager.AlertAbout(target, true)
  81. target.nickTimer.Touch(rb)
  82. } // else: these will be deferred to the end of registration (see #572)
  83. return nil
  84. }
  85. func (server *Server) RandomlyRename(client *Client) {
  86. format := server.Config().Accounts.NickReservation.GuestFormat
  87. buf := make([]byte, 8)
  88. rand.Read(buf)
  89. nick := strings.Replace(format, "*", utils.B32Encoder.EncodeToString(buf), -1)
  90. sessions := client.Sessions()
  91. if len(sessions) == 0 {
  92. // this can happen if they are anonymous and BRB (in general, an always-on
  93. // client has title to its nickname and will never be the victim of
  94. // a call to RandomlyRename)
  95. client.destroy(nil)
  96. return
  97. }
  98. // XXX arbitrarily pick the first session to receive error messages;
  99. // all other sessions receive a `NICK` line same as a friend would
  100. rb := NewResponseBuffer(sessions[0])
  101. performNickChange(server, client, client, nil, nick, rb)
  102. rb.Send(false)
  103. // technically performNickChange can fail to change the nick,
  104. // but if they're still delinquent, the timer will get them later
  105. }
  106. // if force-nick-equals-account is set, account name and nickname must be equal,
  107. // so we need to re-NICK automatically on every login event (IDENTIFY,
  108. // VERIFY, and a REGISTER that auto-verifies). if we can't get the nick
  109. // then we log them out (they will be able to reattach with SASL)
  110. func fixupNickEqualsAccount(client *Client, rb *ResponseBuffer, config *Config) (success bool) {
  111. if !config.Accounts.NickReservation.ForceNickEqualsAccount {
  112. return true
  113. }
  114. if !client.registered {
  115. return true
  116. }
  117. if performNickChange(client.server, client, client, rb.session, client.AccountName(), rb) != nil {
  118. client.server.accounts.Logout(client)
  119. nsNotice(rb, client.t("A client is already using that account; try logging out and logging back in with SASL"))
  120. return false
  121. }
  122. return true
  123. }