mirror of
https://github.com/ergochat/ergo.git
synced 2025-01-08 19:22:53 +01:00
fix #1027
This commit is contained in:
parent
b5afa91e3e
commit
32ee9440c8
@ -873,8 +873,6 @@ func (am *AccountManager) Verify(client *Client, account string, code string) er
|
|||||||
}
|
}
|
||||||
if method == NickEnforcementStrict {
|
if method == NickEnforcementStrict {
|
||||||
am.server.RandomlyRename(currentClient)
|
am.server.RandomlyRename(currentClient)
|
||||||
} else if method == NickEnforcementWithTimeout {
|
|
||||||
currentClient.nickTimer.Touch(nil)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@ -1738,8 +1736,6 @@ func (am *AccountManager) applyVhostToClients(account string, result VHostInfo)
|
|||||||
func (am *AccountManager) Login(client *Client, account ClientAccount) {
|
func (am *AccountManager) Login(client *Client, account ClientAccount) {
|
||||||
client.Login(account)
|
client.Login(account)
|
||||||
|
|
||||||
client.nickTimer.Touch(nil)
|
|
||||||
|
|
||||||
am.applyVHostInfo(client, account.VHost)
|
am.applyVHostInfo(client, account.VHost)
|
||||||
|
|
||||||
casefoldedAccount := client.Account()
|
casefoldedAccount := client.Account()
|
||||||
@ -1757,7 +1753,7 @@ func (am *AccountManager) Logout(client *Client) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
am.logoutOfAccount(client)
|
client.Logout()
|
||||||
|
|
||||||
clients := am.accountToClients[casefoldedAccount]
|
clients := am.accountToClients[casefoldedAccount]
|
||||||
if len(clients) <= 1 {
|
if len(clients) <= 1 {
|
||||||
@ -1934,15 +1930,3 @@ type rawClientAccount struct {
|
|||||||
VHost string
|
VHost string
|
||||||
Settings string
|
Settings string
|
||||||
}
|
}
|
||||||
|
|
||||||
// logoutOfAccount logs the client out of their current account.
|
|
||||||
// TODO(#1027) delete this entire method and just use client.Logout()
|
|
||||||
func (am *AccountManager) logoutOfAccount(client *Client) {
|
|
||||||
if client.Account() == "" {
|
|
||||||
// already logged out
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
client.Logout()
|
|
||||||
go client.nickTimer.Touch(nil)
|
|
||||||
}
|
|
||||||
|
@ -67,7 +67,6 @@ type Client struct {
|
|||||||
nickCasefolded string
|
nickCasefolded string
|
||||||
nickMaskCasefolded string
|
nickMaskCasefolded string
|
||||||
nickMaskString string // cache for nickmask string since it's used with lots of replies
|
nickMaskString string // cache for nickmask string since it's used with lots of replies
|
||||||
nickTimer NickTimer
|
|
||||||
oper *Oper
|
oper *Oper
|
||||||
preregNick string
|
preregNick string
|
||||||
proxiedIP net.IP // actual remote IP if using the PROXY protocol
|
proxiedIP net.IP // actual remote IP if using the PROXY protocol
|
||||||
@ -604,9 +603,6 @@ func (client *Client) run(session *Session) {
|
|||||||
} else {
|
} else {
|
||||||
client.playReattachMessages(session)
|
client.playReattachMessages(session)
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// don't reset the nick timer during a reattach
|
|
||||||
client.nickTimer.Initialize(client)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
firstLine := !isReattach
|
firstLine := !isReattach
|
||||||
@ -1353,7 +1349,6 @@ func (client *Client) destroy(session *Session) {
|
|||||||
client.server.clients.Remove(client)
|
client.server.clients.Remove(client)
|
||||||
|
|
||||||
// clean up self
|
// clean up self
|
||||||
client.nickTimer.Stop()
|
|
||||||
client.brbTimer.Disable()
|
client.brbTimer.Disable()
|
||||||
|
|
||||||
client.server.accounts.Logout(client)
|
client.server.accounts.Logout(client)
|
||||||
|
@ -320,7 +320,6 @@ const (
|
|||||||
// else be fixed up by a schema change)
|
// else be fixed up by a schema change)
|
||||||
NickEnforcementOptional NickEnforcementMethod = iota
|
NickEnforcementOptional NickEnforcementMethod = iota
|
||||||
NickEnforcementNone
|
NickEnforcementNone
|
||||||
NickEnforcementWithTimeout
|
|
||||||
NickEnforcementStrict
|
NickEnforcementStrict
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -330,8 +329,6 @@ func nickReservationToString(method NickEnforcementMethod) string {
|
|||||||
return "default"
|
return "default"
|
||||||
case NickEnforcementNone:
|
case NickEnforcementNone:
|
||||||
return "none"
|
return "none"
|
||||||
case NickEnforcementWithTimeout:
|
|
||||||
return "timeout"
|
|
||||||
case NickEnforcementStrict:
|
case NickEnforcementStrict:
|
||||||
return "strict"
|
return "strict"
|
||||||
default:
|
default:
|
||||||
@ -347,8 +344,6 @@ func nickReservationFromString(method string) (NickEnforcementMethod, error) {
|
|||||||
return NickEnforcementOptional, nil
|
return NickEnforcementOptional, nil
|
||||||
case "none":
|
case "none":
|
||||||
return NickEnforcementNone, nil
|
return NickEnforcementNone, nil
|
||||||
case "timeout":
|
|
||||||
return NickEnforcementWithTimeout, nil
|
|
||||||
case "strict":
|
case "strict":
|
||||||
return NickEnforcementStrict, nil
|
return NickEnforcementStrict, nil
|
||||||
default:
|
default:
|
||||||
|
@ -23,7 +23,7 @@ const (
|
|||||||
// 'version' of the database schema
|
// 'version' of the database schema
|
||||||
keySchemaVersion = "db.version"
|
keySchemaVersion = "db.version"
|
||||||
// latest schema of the db
|
// latest schema of the db
|
||||||
latestDbSchema = "11"
|
latestDbSchema = "12"
|
||||||
|
|
||||||
keyCloakSecret = "crypto.cloak_secret"
|
keyCloakSecret = "crypto.cloak_secret"
|
||||||
)
|
)
|
||||||
@ -650,6 +650,42 @@ func schemaChangeV10ToV11(config *Config, tx *buntdb.Tx) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// #1027: NickEnforcementTimeout (2) was removed,
|
||||||
|
// NickEnforcementStrict was 3 and is now 2
|
||||||
|
func schemaChangeV11ToV12(config *Config, tx *buntdb.Tx) error {
|
||||||
|
prefix := "account.settings "
|
||||||
|
var accounts, rawSettings []string
|
||||||
|
tx.AscendGreaterOrEqual("", prefix, func(key, value string) bool {
|
||||||
|
if !strings.HasPrefix(key, prefix) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
account := strings.TrimPrefix(key, prefix)
|
||||||
|
accounts = append(accounts, account)
|
||||||
|
rawSettings = append(rawSettings, value)
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
|
||||||
|
for i, account := range accounts {
|
||||||
|
var settings AccountSettings
|
||||||
|
err := json.Unmarshal([]byte(rawSettings[i]), &settings)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("corrupt account settings entry for %s: %v\n", account, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// upgrade NickEnforcementTimeout (which was 2) to NickEnforcementStrict (currently 2),
|
||||||
|
// fix up the old value of NickEnforcementStrict (3) to the current value (2)
|
||||||
|
if int(settings.NickEnforcement) == 3 {
|
||||||
|
settings.NickEnforcement = NickEnforcementMethod(2)
|
||||||
|
text, err := json.Marshal(settings)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
tx.Set(prefix+account, string(text), nil)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
allChanges := []SchemaChange{
|
allChanges := []SchemaChange{
|
||||||
{
|
{
|
||||||
@ -702,6 +738,11 @@ func init() {
|
|||||||
TargetVersion: "11",
|
TargetVersion: "11",
|
||||||
Changer: schemaChangeV10ToV11,
|
Changer: schemaChangeV10ToV11,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
InitialVersion: "11",
|
||||||
|
TargetVersion: "12",
|
||||||
|
Changer: schemaChangeV11ToV12,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// build the index
|
// build the index
|
||||||
|
130
irc/idletimer.go
130
irc/idletimer.go
@ -6,10 +6,8 @@ package irc
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/goshuirc/irc-go/ircfmt"
|
|
||||||
"github.com/oragono/oragono/irc/caps"
|
"github.com/oragono/oragono/irc/caps"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -179,134 +177,6 @@ func (it *IdleTimer) quitMessage(state TimerState) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// NickTimer manages timing out of clients who are squatting reserved nicks
|
|
||||||
type NickTimer struct {
|
|
||||||
sync.Mutex // tier 1
|
|
||||||
|
|
||||||
// immutable after construction
|
|
||||||
client *Client
|
|
||||||
|
|
||||||
// mutable
|
|
||||||
nick string
|
|
||||||
accountForNick string
|
|
||||||
account string
|
|
||||||
timeout time.Duration
|
|
||||||
timer *time.Timer
|
|
||||||
enabled uint32
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize sets up a NickTimer, based on server config settings.
|
|
||||||
func (nt *NickTimer) Initialize(client *Client) {
|
|
||||||
if nt.client == nil {
|
|
||||||
nt.client = client // placate the race detector
|
|
||||||
}
|
|
||||||
|
|
||||||
config := &client.server.Config().Accounts.NickReservation
|
|
||||||
enabled := config.Enabled && (config.Method == NickEnforcementWithTimeout || config.AllowCustomEnforcement)
|
|
||||||
|
|
||||||
nt.Lock()
|
|
||||||
defer nt.Unlock()
|
|
||||||
nt.timeout = config.RenameTimeout
|
|
||||||
if enabled {
|
|
||||||
atomic.StoreUint32(&nt.enabled, 1)
|
|
||||||
} else {
|
|
||||||
nt.stopInternal()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (nt *NickTimer) Enabled() bool {
|
|
||||||
return atomic.LoadUint32(&nt.enabled) == 1
|
|
||||||
}
|
|
||||||
|
|
||||||
func (nt *NickTimer) Timeout() (timeout time.Duration) {
|
|
||||||
nt.Lock()
|
|
||||||
timeout = nt.timeout
|
|
||||||
nt.Unlock()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Touch records a nick change and updates the timer as necessary
|
|
||||||
func (nt *NickTimer) Touch(rb *ResponseBuffer) {
|
|
||||||
if !nt.Enabled() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var session *Session
|
|
||||||
if rb != nil {
|
|
||||||
session = rb.session
|
|
||||||
}
|
|
||||||
|
|
||||||
cfnick, skeleton := nt.client.uniqueIdentifiers()
|
|
||||||
account := nt.client.Account()
|
|
||||||
accountForNick, method := nt.client.server.accounts.EnforcementStatus(cfnick, skeleton)
|
|
||||||
enforceTimeout := method == NickEnforcementWithTimeout
|
|
||||||
|
|
||||||
var shouldWarn, shouldRename bool
|
|
||||||
|
|
||||||
func() {
|
|
||||||
nt.Lock()
|
|
||||||
defer nt.Unlock()
|
|
||||||
|
|
||||||
// the timer will not reset as long as the squatter is targeting the same account
|
|
||||||
accountChanged := accountForNick != nt.accountForNick
|
|
||||||
// change state
|
|
||||||
nt.nick = cfnick
|
|
||||||
nt.account = account
|
|
||||||
nt.accountForNick = accountForNick
|
|
||||||
delinquent := accountForNick != "" && accountForNick != account
|
|
||||||
|
|
||||||
if nt.timer != nil && (!enforceTimeout || !delinquent || accountChanged) {
|
|
||||||
nt.timer.Stop()
|
|
||||||
nt.timer = nil
|
|
||||||
}
|
|
||||||
if enforceTimeout && delinquent && (accountChanged || nt.timer == nil) {
|
|
||||||
nt.timer = time.AfterFunc(nt.timeout, nt.processTimeout)
|
|
||||||
shouldWarn = true
|
|
||||||
} else if method == NickEnforcementStrict && delinquent {
|
|
||||||
shouldRename = true // this can happen if reservation was enabled by rehash
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
if shouldWarn {
|
|
||||||
tnick := nt.client.Nick()
|
|
||||||
message := fmt.Sprintf(ircfmt.Unescape(nt.client.t(nsTimeoutNotice)), nt.Timeout())
|
|
||||||
// #449
|
|
||||||
for _, mSession := range nt.client.Sessions() {
|
|
||||||
if mSession == session {
|
|
||||||
rb.Add(nil, nsPrefix, "NOTICE", tnick, message)
|
|
||||||
rb.Add(nil, nt.client.server.name, "WARN", "*", "ACCOUNT_REQUIRED", message)
|
|
||||||
} else {
|
|
||||||
mSession.Send(nil, nsPrefix, "NOTICE", tnick, message)
|
|
||||||
mSession.Send(nil, nt.client.server.name, "WARN", "*", "ACCOUNT_REQUIRED", message)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if shouldRename {
|
|
||||||
nt.client.Notice(nt.client.t("Nickname is reserved by a different account"))
|
|
||||||
nt.client.server.RandomlyRename(nt.client)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stop stops counting time and cleans up the timer
|
|
||||||
func (nt *NickTimer) Stop() {
|
|
||||||
nt.Lock()
|
|
||||||
defer nt.Unlock()
|
|
||||||
nt.stopInternal()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (nt *NickTimer) stopInternal() {
|
|
||||||
if nt.timer != nil {
|
|
||||||
nt.timer.Stop()
|
|
||||||
nt.timer = nil
|
|
||||||
}
|
|
||||||
atomic.StoreUint32(&nt.enabled, 0)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (nt *NickTimer) processTimeout() {
|
|
||||||
baseMsg := "Nick is reserved and authentication timeout expired: %v"
|
|
||||||
nt.client.Notice(fmt.Sprintf(nt.client.t(baseMsg), nt.Timeout()))
|
|
||||||
nt.client.server.RandomlyRename(nt.client)
|
|
||||||
}
|
|
||||||
|
|
||||||
// BrbTimer is a timer on the client as a whole (not an individual session) for implementing
|
// BrbTimer is a timer on the client as a whole (not an individual session) for implementing
|
||||||
// the BRB command and related functionality (where a client can remain online without
|
// the BRB command and related functionality (where a client can remain online without
|
||||||
// having any connected sessions).
|
// having any connected sessions).
|
||||||
|
@ -97,7 +97,6 @@ func performNickChange(server *Server, client *Client, target *Client, session *
|
|||||||
client.server.monitorManager.AlertAbout(details.nick, details.nickCasefolded, false)
|
client.server.monitorManager.AlertAbout(details.nick, details.nickCasefolded, false)
|
||||||
client.server.monitorManager.AlertAbout(assignedNickname, newCfnick, true)
|
client.server.monitorManager.AlertAbout(assignedNickname, newCfnick, true)
|
||||||
}
|
}
|
||||||
target.nickTimer.Touch(rb)
|
|
||||||
} // else: these will be deferred to the end of registration (see #572)
|
} // else: these will be deferred to the end of registration (see #572)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -37,11 +37,6 @@ func servCmdRequiresBouncerEnabled(config *Config) bool {
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
nsPrefix = "NickServ!NickServ@localhost"
|
nsPrefix = "NickServ!NickServ@localhost"
|
||||||
// ZNC's nickserv module will not detect this unless it is:
|
|
||||||
// 1. sent with prefix `nickserv`
|
|
||||||
// 2. contains the string "identify"
|
|
||||||
// 3. contains at least one of several other magic strings ("msg" works)
|
|
||||||
nsTimeoutNotice = `This nickname is reserved. Please login within %v (using $b/msg NickServ IDENTIFY <password>$b or SASL), or switch to a different nickname.`
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const nickservHelp = `NickServ lets you register, log in to, and manage an account.`
|
const nickservHelp = `NickServ lets you register, log in to, and manage an account.`
|
||||||
@ -249,10 +244,9 @@ SET modifies your account settings. The following settings are available:`,
|
|||||||
'enforce' lets you specify a custom enforcement mechanism for your registered
|
'enforce' lets you specify a custom enforcement mechanism for your registered
|
||||||
nicknames. Your options are:
|
nicknames. Your options are:
|
||||||
1. 'none' [no enforcement, overriding the server default]
|
1. 'none' [no enforcement, overriding the server default]
|
||||||
2. 'timeout' [anyone using the nick must authenticate before a deadline,
|
|
||||||
or else they will be renamed]
|
or else they will be renamed]
|
||||||
3. 'strict' [you must already be authenticated to use the nick]
|
2. 'strict' [you must already be authenticated to use the nick]
|
||||||
4. 'default' [use the server default]`,
|
3. 'default' [use the server default]`,
|
||||||
|
|
||||||
`$bMULTICLIENT$b
|
`$bMULTICLIENT$b
|
||||||
If 'multiclient' is enabled and you are already logged in and using a nick, a
|
If 'multiclient' is enabled and you are already logged in and using a nick, a
|
||||||
|
@ -320,9 +320,6 @@ func (server *Server) playRegistrationBurst(session *Session) {
|
|||||||
if server.logger.IsLoggingRawIO() {
|
if server.logger.IsLoggingRawIO() {
|
||||||
session.Send(nil, c.server.name, "NOTICE", d.nick, c.t("This server is in debug mode and is logging all user I/O. If you do not wish for everything you send to be readable by the server owner(s), please disconnect."))
|
session.Send(nil, c.server.name, "NOTICE", d.nick, c.t("This server is in debug mode and is logging all user I/O. If you do not wish for everything you send to be readable by the server owner(s), please disconnect."))
|
||||||
}
|
}
|
||||||
|
|
||||||
// #572: defer nick warnings to the end of the registration burst
|
|
||||||
session.client.nickTimer.Touch(nil)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// RplISupport outputs our ISUPPORT lines to the client. This is used on connection and in VERSION responses.
|
// RplISupport outputs our ISUPPORT lines to the client. This is used on connection and in VERSION responses.
|
||||||
@ -629,13 +626,6 @@ func (server *Server) applyConfig(config *Config) (err error) {
|
|||||||
if sendRawOutputNotice {
|
if sendRawOutputNotice {
|
||||||
sClient.Notice(sClient.t("This server is in debug mode and is logging all user I/O. If you do not wish for everything you send to be readable by the server owner(s), please disconnect."))
|
sClient.Notice(sClient.t("This server is in debug mode and is logging all user I/O. If you do not wish for everything you send to be readable by the server owner(s), please disconnect."))
|
||||||
}
|
}
|
||||||
|
|
||||||
if !oldConfig.Accounts.NickReservation.Enabled && config.Accounts.NickReservation.Enabled {
|
|
||||||
sClient.nickTimer.Initialize(sClient)
|
|
||||||
sClient.nickTimer.Touch(nil)
|
|
||||||
} else if oldConfig.Accounts.NickReservation.Enabled && !config.Accounts.NickReservation.Enabled {
|
|
||||||
sClient.nickTimer.Stop()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user