2017-10-02 10:42:50 +02:00
|
|
|
// Copyright (c) 2017 Shivaram Lingamneni <slingamn@cs.stanford.edu>
|
|
|
|
// released under the MIT license
|
|
|
|
|
|
|
|
package irc
|
|
|
|
|
2018-02-03 11:21:32 +01:00
|
|
|
import (
|
2022-05-20 07:32:39 +02:00
|
|
|
"fmt"
|
2023-08-16 02:57:52 +02:00
|
|
|
"maps"
|
2019-05-08 10:11:54 +02:00
|
|
|
"net"
|
2019-03-12 00:24:45 +01:00
|
|
|
"time"
|
|
|
|
|
2021-05-25 06:34:38 +02:00
|
|
|
"github.com/ergochat/ergo/irc/caps"
|
|
|
|
"github.com/ergochat/ergo/irc/languages"
|
|
|
|
"github.com/ergochat/ergo/irc/modes"
|
|
|
|
"github.com/ergochat/ergo/irc/utils"
|
2025-01-14 03:47:21 +01:00
|
|
|
"github.com/ergochat/ergo/irc/webpush"
|
2018-02-03 11:21:32 +01:00
|
|
|
)
|
2017-10-05 15:39:57 +02:00
|
|
|
|
2019-02-19 08:54:57 +01:00
|
|
|
func (server *Server) Config() (config *Config) {
|
2022-08-03 06:59:00 +02:00
|
|
|
return server.config.Load()
|
2018-03-18 02:32:12 +01:00
|
|
|
}
|
|
|
|
|
2018-04-19 08:48:19 +02:00
|
|
|
func (server *Server) GetOperator(name string) (oper *Oper) {
|
|
|
|
name, err := CasefoldName(name)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2019-05-09 20:18:30 +02:00
|
|
|
return server.Config().operators[name]
|
2018-04-19 08:48:19 +02:00
|
|
|
}
|
|
|
|
|
2019-02-19 08:54:57 +01:00
|
|
|
func (server *Server) Languages() (lm *languages.Manager) {
|
|
|
|
return server.Config().languageManager
|
|
|
|
}
|
|
|
|
|
2020-07-08 11:32:14 +02:00
|
|
|
func (server *Server) Defcon() uint32 {
|
2022-08-10 08:47:39 +02:00
|
|
|
return server.defcon.Load()
|
2020-07-08 11:32:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (server *Server) SetDefcon(defcon uint32) {
|
2022-08-10 08:47:39 +02:00
|
|
|
server.defcon.Store(defcon)
|
2020-07-08 11:32:14 +02:00
|
|
|
}
|
|
|
|
|
2019-04-12 06:08:46 +02:00
|
|
|
func (client *Client) Sessions() (sessions []*Session) {
|
|
|
|
client.stateMutex.RLock()
|
2019-08-02 06:20:31 +02:00
|
|
|
sessions = client.sessions
|
2019-04-12 06:08:46 +02:00
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-05-08 10:11:54 +02:00
|
|
|
type SessionData struct {
|
2020-09-19 19:19:41 +02:00
|
|
|
ctime time.Time
|
|
|
|
atime time.Time
|
|
|
|
ip net.IP
|
|
|
|
hostname string
|
|
|
|
certfp string
|
|
|
|
deviceID string
|
2020-09-24 08:44:12 +02:00
|
|
|
connInfo string
|
2025-01-12 05:07:04 +01:00
|
|
|
connID string
|
2020-09-19 19:19:41 +02:00
|
|
|
sessionID int64
|
2021-03-18 09:24:45 +01:00
|
|
|
caps []string
|
2019-05-08 10:11:54 +02:00
|
|
|
}
|
|
|
|
|
2020-09-24 14:35:03 +02:00
|
|
|
func (client *Client) AllSessionData(currentSession *Session, hasPrivs bool) (data []SessionData, currentIndex int) {
|
2019-05-08 10:11:54 +02:00
|
|
|
currentIndex = -1
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
|
|
|
|
data = make([]SessionData, len(client.sessions))
|
|
|
|
for i, session := range client.sessions {
|
|
|
|
if session == currentSession {
|
|
|
|
currentIndex = i
|
|
|
|
}
|
|
|
|
data[i] = SessionData{
|
2020-09-19 19:19:41 +02:00
|
|
|
atime: session.lastActive,
|
|
|
|
ctime: session.ctime,
|
|
|
|
hostname: session.rawHostname,
|
|
|
|
certfp: session.certfp,
|
|
|
|
deviceID: session.deviceID,
|
2025-01-12 05:07:04 +01:00
|
|
|
connID: session.connID,
|
2020-09-19 19:19:41 +02:00
|
|
|
sessionID: session.sessionID,
|
2019-05-08 10:11:54 +02:00
|
|
|
}
|
|
|
|
if session.proxiedIP != nil {
|
|
|
|
data[i].ip = session.proxiedIP
|
|
|
|
} else {
|
|
|
|
data[i].ip = session.realIP
|
|
|
|
}
|
2020-09-24 14:35:03 +02:00
|
|
|
if hasPrivs {
|
|
|
|
data[i].connInfo = utils.DescribeConn(session.socket.conn.UnderlyingConn().Conn)
|
|
|
|
}
|
2021-03-18 09:24:45 +01:00
|
|
|
data[i].caps = session.capabilities.Strings(caps.Cap302, nil, 300)
|
2019-05-08 10:11:54 +02:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-02-05 06:50:14 +01:00
|
|
|
func (client *Client) AddSession(session *Session) (success bool, numSessions int, lastSeen time.Time, wasAway, nowAway string) {
|
2021-03-18 07:53:18 +01:00
|
|
|
config := client.server.Config()
|
2019-04-12 06:08:46 +02:00
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
2019-05-22 03:40:25 +02:00
|
|
|
// client may be dying and ineligible to receive another session
|
2019-05-28 10:56:56 +02:00
|
|
|
if client.destroyed {
|
2020-02-19 01:38:42 +01:00
|
|
|
return
|
2019-04-12 06:08:46 +02:00
|
|
|
}
|
2019-05-22 03:40:25 +02:00
|
|
|
// success, attach the new session to the client
|
2019-04-12 06:08:46 +02:00
|
|
|
session.client = client
|
2020-09-19 19:19:41 +02:00
|
|
|
session.sessionID = client.nextSessionID
|
|
|
|
client.nextSessionID++
|
2019-08-02 06:20:31 +02:00
|
|
|
newSessions := make([]*Session, len(client.sessions)+1)
|
|
|
|
copy(newSessions, client.sessions)
|
|
|
|
newSessions[len(newSessions)-1] = session
|
2020-06-18 09:38:00 +02:00
|
|
|
if client.accountSettings.AutoreplayMissed || session.deviceID != "" {
|
2020-06-12 21:51:48 +02:00
|
|
|
lastSeen = client.lastSeen[session.deviceID]
|
2020-06-27 00:02:22 +02:00
|
|
|
client.setLastSeen(time.Now().UTC(), session.deviceID)
|
2020-02-19 01:38:42 +01:00
|
|
|
}
|
2019-08-02 06:20:31 +02:00
|
|
|
client.sessions = newSessions
|
2023-02-05 06:50:14 +01:00
|
|
|
wasAway = client.awayMessage
|
|
|
|
if client.autoAwayEnabledNoMutex(config) {
|
|
|
|
client.setAutoAwayNoMutex(config)
|
|
|
|
} else {
|
|
|
|
if session.awayMessage != "" && session.awayMessage != "*" {
|
|
|
|
// set the away message
|
|
|
|
client.awayMessage = session.awayMessage
|
|
|
|
} else if session.awayMessage == "" && !session.awayAt.IsZero() {
|
|
|
|
// weird edge case: explicit `AWAY` or `AWAY :` during pre-registration makes the client back
|
|
|
|
client.awayMessage = ""
|
2021-03-18 07:53:18 +01:00
|
|
|
}
|
2023-02-05 06:50:14 +01:00
|
|
|
// else: the client sent no AWAY command at all, no-op
|
|
|
|
// or: the client sent `AWAY *`, which should not modify the publicly visible away state
|
2020-05-19 20:12:20 +02:00
|
|
|
}
|
2023-02-05 06:50:14 +01:00
|
|
|
nowAway = client.awayMessage
|
|
|
|
return true, len(client.sessions), lastSeen, wasAway, nowAway
|
2019-04-12 06:08:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) removeSession(session *Session) (success bool, length int) {
|
|
|
|
if len(client.sessions) == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
sessions := make([]*Session, 0, len(client.sessions)-1)
|
|
|
|
for _, currentSession := range client.sessions {
|
|
|
|
if session == currentSession {
|
|
|
|
success = true
|
|
|
|
} else {
|
|
|
|
sessions = append(sessions, currentSession)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
client.sessions = sessions
|
|
|
|
length = len(sessions)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-26 04:31:38 +02:00
|
|
|
// #1650: show an arbitrarily chosen session IP and hostname in RPL_WHOISACTUALLY
|
|
|
|
func (client *Client) getWhoisActually() (ip net.IP, hostname string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
|
|
|
|
for _, session := range client.sessions {
|
|
|
|
return session.IP(), session.rawHostname
|
|
|
|
}
|
|
|
|
return utils.IPv4LoopbackAddress, client.server.name
|
|
|
|
}
|
|
|
|
|
2017-11-03 07:36:55 +01:00
|
|
|
func (client *Client) Nick() string {
|
2017-10-04 06:57:03 +02:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.nick
|
|
|
|
}
|
|
|
|
|
2017-11-03 07:36:55 +01:00
|
|
|
func (client *Client) NickMaskString() string {
|
2017-10-04 06:57:03 +02:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.nickMaskString
|
|
|
|
}
|
|
|
|
|
2017-11-03 07:36:55 +01:00
|
|
|
func (client *Client) NickCasefolded() string {
|
2017-10-04 06:57:03 +02:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.nickCasefolded
|
|
|
|
}
|
2017-10-15 18:24:28 +02:00
|
|
|
|
2018-12-23 19:25:02 +01:00
|
|
|
func (client *Client) NickMaskCasefolded() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.nickMaskCasefolded
|
|
|
|
}
|
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
func (client *Client) Username() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.username
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) Hostname() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.hostname
|
|
|
|
}
|
|
|
|
|
2020-07-17 07:55:13 +02:00
|
|
|
func (client *Client) Away() (result bool, message string) {
|
2019-04-28 21:10:03 +02:00
|
|
|
client.stateMutex.Lock()
|
2021-03-18 07:53:18 +01:00
|
|
|
message = client.awayMessage
|
2019-04-28 21:10:03 +02:00
|
|
|
client.stateMutex.Unlock()
|
2021-03-18 07:53:18 +01:00
|
|
|
result = client.awayMessage != ""
|
2019-04-28 21:10:03 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-02-05 06:50:14 +01:00
|
|
|
func (session *Session) SetAway(awayMessage string) (wasAway, nowAway string) {
|
2021-03-18 07:53:18 +01:00
|
|
|
client := session.client
|
|
|
|
config := client.server.Config()
|
|
|
|
|
2019-04-28 21:10:03 +02:00
|
|
|
client.stateMutex.Lock()
|
2021-03-18 07:53:18 +01:00
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
session.awayMessage = awayMessage
|
|
|
|
session.awayAt = time.Now().UTC()
|
|
|
|
|
2023-02-05 06:50:14 +01:00
|
|
|
wasAway = client.awayMessage
|
|
|
|
if client.autoAwayEnabledNoMutex(config) {
|
2021-03-18 07:53:18 +01:00
|
|
|
client.setAutoAwayNoMutex(config)
|
2023-02-05 06:50:14 +01:00
|
|
|
} else if awayMessage != "*" {
|
2021-03-18 07:53:18 +01:00
|
|
|
client.awayMessage = awayMessage
|
2023-02-05 06:50:14 +01:00
|
|
|
} // else: `AWAY *`, should not modify publicly visible away state
|
|
|
|
nowAway = client.awayMessage
|
2019-04-28 21:10:03 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-02-05 06:50:14 +01:00
|
|
|
func (client *Client) autoAwayEnabledNoMutex(config *Config) bool {
|
|
|
|
return client.registered && client.alwaysOn &&
|
|
|
|
persistenceEnabled(config.Accounts.Multiclient.AutoAway, client.accountSettings.AutoAway)
|
|
|
|
}
|
|
|
|
|
2021-03-18 07:53:18 +01:00
|
|
|
func (client *Client) setAutoAwayNoMutex(config *Config) {
|
|
|
|
// aggregate the away statuses of the individual sessions:
|
|
|
|
var globalAwayState string
|
|
|
|
var awaySetAt time.Time
|
|
|
|
for _, cSession := range client.sessions {
|
|
|
|
if cSession.awayMessage == "" {
|
|
|
|
// a session is active, we are not auto-away
|
|
|
|
client.awayMessage = ""
|
|
|
|
return
|
2023-02-05 06:50:14 +01:00
|
|
|
} else if cSession.awayAt.After(awaySetAt) && cSession.awayMessage != "*" {
|
|
|
|
// choose the latest valid away message from any session
|
2021-03-18 07:53:18 +01:00
|
|
|
globalAwayState = cSession.awayMessage
|
|
|
|
awaySetAt = cSession.awayAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if awaySetAt.IsZero() {
|
|
|
|
// no sessions, enable auto-away
|
|
|
|
client.awayMessage = config.languageManager.Translate(client.languages, `User is currently disconnected`)
|
|
|
|
} else {
|
|
|
|
client.awayMessage = globalAwayState
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-19 01:38:42 +01:00
|
|
|
func (client *Client) AlwaysOn() (alwaysOn bool) {
|
2020-12-02 09:56:00 +01:00
|
|
|
client.stateMutex.RLock()
|
2020-07-26 21:51:33 +02:00
|
|
|
alwaysOn = client.registered && client.alwaysOn
|
2020-12-02 09:56:00 +01:00
|
|
|
client.stateMutex.RUnlock()
|
2020-02-19 01:38:42 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-31 00:59:49 +01:00
|
|
|
// uniqueIdentifiers returns the strings for which the server enforces per-client
|
|
|
|
// uniqueness/ownership; no two clients can have colliding casefolded nicks or
|
|
|
|
// skeletons.
|
|
|
|
func (client *Client) uniqueIdentifiers() (nickCasefolded string, skeleton string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.nickCasefolded, client.skeleton
|
|
|
|
}
|
|
|
|
|
2018-04-19 08:48:19 +02:00
|
|
|
func (client *Client) Oper() *Oper {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.oper
|
|
|
|
}
|
|
|
|
|
2020-08-06 09:16:58 +02:00
|
|
|
func (client *Client) Registered() (result bool) {
|
2019-02-10 02:01:47 +01:00
|
|
|
// `registered` is only written from the client's own goroutine, but may be
|
|
|
|
// read from other goroutines; therefore, the client's own goroutine may read
|
|
|
|
// the value without synchronization, but must write it with synchronization,
|
|
|
|
// and other goroutines must read it with synchronization
|
2020-08-06 09:16:58 +02:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
result = client.registered
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
2019-02-10 02:01:47 +01:00
|
|
|
}
|
|
|
|
|
2019-02-26 03:50:43 +01:00
|
|
|
func (client *Client) RawHostname() (result string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
result = client.rawHostname
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-02-17 20:29:04 +01:00
|
|
|
func (client *Client) AwayMessage() (result string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
result = client.awayMessage
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-02-11 11:30:40 +01:00
|
|
|
func (client *Client) Account() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.account
|
|
|
|
}
|
|
|
|
|
2017-11-09 04:19:50 +01:00
|
|
|
func (client *Client) AccountName() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
2018-02-11 11:30:40 +01:00
|
|
|
return client.accountName
|
|
|
|
}
|
|
|
|
|
2020-02-19 01:38:42 +01:00
|
|
|
func (client *Client) Login(account ClientAccount) {
|
2020-02-21 05:55:42 +01:00
|
|
|
alwaysOn := persistenceEnabled(client.server.Config().Accounts.Multiclient.AlwaysOn, account.Settings.AlwaysOn)
|
2020-02-19 01:38:42 +01:00
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
client.account = account.NameCasefolded
|
|
|
|
client.accountName = account.Name
|
|
|
|
client.accountSettings = account.Settings
|
2020-07-26 21:51:33 +02:00
|
|
|
// mark always-on here: it will not be respected until the client is registered
|
|
|
|
client.alwaysOn = alwaysOn
|
2020-02-19 01:38:42 +01:00
|
|
|
client.accountRegDate = account.RegisteredAt
|
|
|
|
return
|
|
|
|
}
|
2018-02-20 10:50:46 +01:00
|
|
|
|
2020-11-11 01:59:12 +01:00
|
|
|
func (client *Client) setAccountName(name string) {
|
|
|
|
// XXX this assumes validation elsewhere
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
client.accountName = name
|
|
|
|
}
|
|
|
|
|
2021-01-12 14:40:13 +01:00
|
|
|
func (client *Client) setCloakedHostname(cloak string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
client.cloakedHostname = cloak
|
|
|
|
client.updateNickMaskNoMutex()
|
|
|
|
}
|
|
|
|
|
2021-01-19 14:49:45 +01:00
|
|
|
func (client *Client) CloakedHostname() string {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
return client.cloakedHostname
|
|
|
|
}
|
|
|
|
|
2020-02-19 01:38:42 +01:00
|
|
|
func (client *Client) historyCutoff() (cutoff time.Time) {
|
2018-02-11 11:30:40 +01:00
|
|
|
client.stateMutex.Lock()
|
2020-02-19 01:38:42 +01:00
|
|
|
if client.account != "" {
|
|
|
|
cutoff = client.accountRegDate
|
|
|
|
} else {
|
|
|
|
cutoff = client.ctime
|
|
|
|
}
|
|
|
|
client.stateMutex.Unlock()
|
2018-02-20 10:50:46 +01:00
|
|
|
return
|
2017-11-09 04:19:50 +01:00
|
|
|
}
|
|
|
|
|
2020-02-19 01:38:42 +01:00
|
|
|
func (client *Client) Logout() {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.account = ""
|
2020-04-06 19:10:38 +02:00
|
|
|
client.accountName = "*"
|
2020-02-19 01:38:42 +01:00
|
|
|
client.alwaysOn = false
|
2020-02-21 12:36:32 +01:00
|
|
|
client.accountRegDate = time.Time{}
|
|
|
|
client.accountSettings = AccountSettings{}
|
2020-02-19 01:38:42 +01:00
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-05-19 10:27:44 +02:00
|
|
|
func (client *Client) AccountSettings() (result AccountSettings) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
result = client.accountSettings
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetAccountSettings(settings AccountSettings) {
|
2020-03-02 07:53:02 +01:00
|
|
|
// we mark dirty if the client is transitioning to always-on
|
2020-12-21 11:11:50 +01:00
|
|
|
var becameAlwaysOn bool
|
2020-02-21 05:55:42 +01:00
|
|
|
alwaysOn := persistenceEnabled(client.server.Config().Accounts.Multiclient.AlwaysOn, settings.AlwaysOn)
|
2019-05-19 10:27:44 +02:00
|
|
|
client.stateMutex.Lock()
|
2020-02-19 01:38:42 +01:00
|
|
|
if client.registered {
|
2020-07-27 12:08:06 +02:00
|
|
|
// only allow the client to become always-on if their nick equals their account name
|
|
|
|
alwaysOn = alwaysOn && client.nick == client.accountName
|
2020-06-12 21:51:48 +02:00
|
|
|
becameAlwaysOn = (!client.alwaysOn && alwaysOn)
|
2020-02-19 01:38:42 +01:00
|
|
|
client.alwaysOn = alwaysOn
|
|
|
|
}
|
2020-06-12 21:51:48 +02:00
|
|
|
client.accountSettings = settings
|
2019-05-19 10:27:44 +02:00
|
|
|
client.stateMutex.Unlock()
|
2020-06-12 21:51:48 +02:00
|
|
|
if becameAlwaysOn {
|
2020-03-02 07:53:02 +01:00
|
|
|
client.markDirty(IncludeAllAttrs)
|
|
|
|
}
|
2019-05-19 10:27:44 +02:00
|
|
|
}
|
|
|
|
|
2019-02-19 08:54:57 +01:00
|
|
|
func (client *Client) Languages() (languages []string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
languages = client.languages
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return languages
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetLanguages(languages []string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.languages = languages
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
2018-02-03 11:21:32 +01:00
|
|
|
func (client *Client) HasMode(mode modes.Mode) bool {
|
2018-04-23 00:47:10 +02:00
|
|
|
// client.flags has its own synch
|
2020-02-19 01:38:42 +01:00
|
|
|
return client.modes.HasMode(mode)
|
2018-04-23 00:47:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetMode(mode modes.Mode, on bool) bool {
|
2020-02-19 01:38:42 +01:00
|
|
|
return client.modes.SetMode(mode, on)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetRealname(realname string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.realname = realname
|
2020-07-26 21:51:33 +02:00
|
|
|
alwaysOn := client.registered && client.alwaysOn
|
2020-02-19 01:38:42 +01:00
|
|
|
client.stateMutex.Unlock()
|
2020-07-06 10:08:04 +02:00
|
|
|
if alwaysOn {
|
|
|
|
client.markDirty(IncludeRealname)
|
|
|
|
}
|
2017-10-23 01:50:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) Channels() (result []*Channel) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
length := len(client.channels)
|
|
|
|
result = make([]*Channel, length)
|
|
|
|
i := 0
|
|
|
|
for channel := range client.channels {
|
|
|
|
result[i] = channel
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-02-06 10:55:05 +01:00
|
|
|
func (client *Client) NumChannels() int {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return len(client.channels)
|
|
|
|
}
|
|
|
|
|
2018-05-04 06:24:54 +02:00
|
|
|
func (client *Client) WhoWas() (result WhoWas) {
|
2019-01-01 19:00:16 +01:00
|
|
|
return client.Details().WhoWas
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) Details() (result ClientDetails) {
|
2018-05-04 06:24:54 +02:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
2019-05-09 00:14:49 +02:00
|
|
|
return client.detailsNoMutex()
|
|
|
|
}
|
2018-05-04 06:24:54 +02:00
|
|
|
|
2019-05-09 00:14:49 +02:00
|
|
|
func (client *Client) detailsNoMutex() (result ClientDetails) {
|
2019-01-01 19:00:16 +01:00
|
|
|
result.nick = client.nick
|
|
|
|
result.nickCasefolded = client.nickCasefolded
|
2018-05-04 06:24:54 +02:00
|
|
|
result.username = client.username
|
2019-02-06 20:14:32 +01:00
|
|
|
result.hostname = client.hostname
|
2018-05-04 06:24:54 +02:00
|
|
|
result.realname = client.realname
|
2021-06-20 20:13:18 +02:00
|
|
|
result.ip = client.getIPNoMutex()
|
2019-01-01 19:00:16 +01:00
|
|
|
result.nickMask = client.nickMaskString
|
|
|
|
result.nickMaskCasefolded = client.nickMaskCasefolded
|
|
|
|
result.account = client.account
|
|
|
|
result.accountName = client.accountName
|
2018-05-04 06:24:54 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-06-29 10:32:39 +02:00
|
|
|
func (client *Client) UpdateActive(session *Session) {
|
|
|
|
now := time.Now().UTC()
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
client.lastActive = now
|
|
|
|
session.lastActive = now
|
|
|
|
}
|
|
|
|
|
2020-07-12 22:31:51 +02:00
|
|
|
func (client *Client) Realname() string {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
result := client.realname
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-12-21 11:11:50 +01:00
|
|
|
func (client *Client) IsExpiredAlwaysOn(config *Config) (result bool) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
2021-01-15 12:50:35 +01:00
|
|
|
return client.checkAlwaysOnExpirationNoMutex(config, false)
|
2020-12-21 11:11:50 +01:00
|
|
|
}
|
|
|
|
|
2021-01-15 12:50:35 +01:00
|
|
|
func (client *Client) checkAlwaysOnExpirationNoMutex(config *Config, ignoreRegistration bool) (result bool) {
|
|
|
|
if !((client.registered || ignoreRegistration) && client.alwaysOn) {
|
2020-12-21 11:11:50 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
deadline := time.Duration(config.Accounts.Multiclient.AlwaysOnExpiration)
|
|
|
|
if deadline == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
now := time.Now()
|
|
|
|
for _, ts := range client.lastSeen {
|
|
|
|
if now.Sub(ts) < deadline {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2022-03-30 21:35:28 +02:00
|
|
|
func (client *Client) GetReadMarker(cfname string) (result string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
t, ok := client.readMarkers[cfname]
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
if ok {
|
2022-05-20 07:32:39 +02:00
|
|
|
return fmt.Sprintf("timestamp=%s", t.Format(IRCv3TimestampFormat))
|
2022-03-30 21:35:28 +02:00
|
|
|
}
|
|
|
|
return "*"
|
|
|
|
}
|
|
|
|
|
2025-01-14 03:47:21 +01:00
|
|
|
func (client *Client) getMarkreadTime(cfname string) (timestamp time.Time, ok bool) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
timestamp, ok = client.readMarkers[cfname]
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-30 21:35:28 +02:00
|
|
|
func (client *Client) copyReadMarkers() (result map[string]time.Time) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
2023-08-16 02:57:52 +02:00
|
|
|
return maps.Clone(client.readMarkers)
|
2022-03-30 21:35:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetReadMarker(cfname string, now time.Time) (result time.Time) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
if client.readMarkers == nil {
|
|
|
|
client.readMarkers = make(map[string]time.Time)
|
|
|
|
}
|
|
|
|
result = updateLRUMap(client.readMarkers, cfname, now, maxReadMarkers)
|
|
|
|
client.dirtyTimestamps = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateLRUMap(lru map[string]time.Time, key string, val time.Time, maxItems int) (result time.Time) {
|
|
|
|
if currentVal := lru[key]; currentVal.After(val) {
|
|
|
|
return currentVal
|
|
|
|
}
|
|
|
|
|
|
|
|
lru[key] = val
|
|
|
|
// evict the least-recently-used entry if necessary
|
|
|
|
if maxItems < len(lru) {
|
|
|
|
var minKey string
|
|
|
|
var minVal time.Time
|
|
|
|
for key, val := range lru {
|
|
|
|
if minVal.IsZero() || val.Before(minVal) {
|
|
|
|
minKey, minVal = key, val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete(lru, minKey)
|
|
|
|
}
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
|
2025-01-14 03:47:21 +01:00
|
|
|
func (client *Client) addClearablePushMessage(cftarget string, messageTime time.Time) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
if client.clearablePushMessages == nil {
|
|
|
|
client.clearablePushMessages = make(map[string]time.Time)
|
|
|
|
}
|
|
|
|
updateLRUMap(client.clearablePushMessages, cftarget, messageTime, maxReadMarkers)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) clearClearablePushMessage(cftarget string, readTimestamp time.Time) (ok bool) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
pushMessageTime, ok := client.clearablePushMessages[cftarget]
|
|
|
|
if ok && utils.ReadMarkerLessThanOrEqual(pushMessageTime, readTimestamp) {
|
|
|
|
delete(client.clearablePushMessages, cftarget)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-03-30 21:35:28 +02:00
|
|
|
func (client *Client) shouldFlushTimestamps() (result bool) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
result = client.dirtyTimestamps && client.registered && client.alwaysOn
|
|
|
|
client.dirtyTimestamps = false
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-05-03 18:46:12 +02:00
|
|
|
func (client *Client) setKlined() {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.isKlined = true
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
2025-01-14 03:47:21 +01:00
|
|
|
func (client *Client) refreshPushSubscription(endpoint string, keys webpush.Keys) bool {
|
|
|
|
// do not mark dirty --- defer the write to periodic maintenance
|
|
|
|
now := time.Now().UTC()
|
|
|
|
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
sub, ok := client.pushSubscriptions[endpoint]
|
|
|
|
if ok && sub.Keys.Equal(keys) {
|
|
|
|
sub.LastRefresh = now
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false // subscription doesn't exist, we need to send a test message
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) addPushSubscription(endpoint string, keys webpush.Keys) error {
|
|
|
|
changed := false
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if changed {
|
|
|
|
client.markDirty(IncludeAllAttrs)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
config := client.server.Config()
|
|
|
|
now := time.Now().UTC()
|
|
|
|
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
if client.pushSubscriptions == nil {
|
|
|
|
client.pushSubscriptions = make(map[string]*pushSubscription)
|
|
|
|
}
|
|
|
|
|
|
|
|
sub, ok := client.pushSubscriptions[endpoint]
|
|
|
|
if ok {
|
|
|
|
changed = !sub.Keys.Equal(keys)
|
|
|
|
sub.Keys = keys
|
|
|
|
sub.LastRefresh = now
|
|
|
|
} else {
|
|
|
|
if len(client.pushSubscriptions) >= config.WebPush.MaxSubscriptions {
|
|
|
|
return errLimitExceeded
|
|
|
|
}
|
|
|
|
changed = true
|
|
|
|
sub = newPushSubscription(storedPushSubscription{
|
|
|
|
Endpoint: endpoint,
|
|
|
|
Keys: keys,
|
|
|
|
LastRefresh: now,
|
|
|
|
LastSuccess: now, // assume we just sent a successful message to confirm the sub
|
|
|
|
})
|
|
|
|
client.pushSubscriptions[endpoint] = sub
|
|
|
|
}
|
|
|
|
|
|
|
|
if changed {
|
|
|
|
client.rebuildPushSubscriptionCache()
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) hasPushSubscriptions() bool {
|
|
|
|
return client.pushSubscriptionsExist.Load() != 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) getPushSubscriptions() []storedPushSubscription {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
|
|
|
|
return client.cachedPushSubscriptions
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) rebuildPushSubscriptionCache() {
|
|
|
|
// must hold write lock
|
|
|
|
if len(client.pushSubscriptions) == 0 {
|
|
|
|
client.cachedPushSubscriptions = nil
|
|
|
|
client.pushSubscriptionsExist.Store(0)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
client.cachedPushSubscriptions = make([]storedPushSubscription, 0, len(client.pushSubscriptions))
|
|
|
|
for _, subscription := range client.pushSubscriptions {
|
|
|
|
client.cachedPushSubscriptions = append(client.cachedPushSubscriptions, subscription.storedPushSubscription)
|
|
|
|
}
|
|
|
|
client.pushSubscriptionsExist.Store(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) deletePushSubscription(endpoint string, writeback bool) (changed bool) {
|
|
|
|
defer func() {
|
|
|
|
if writeback && changed {
|
|
|
|
client.markDirty(IncludeAllAttrs)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
_, ok := client.pushSubscriptions[endpoint]
|
|
|
|
if ok {
|
|
|
|
changed = true
|
|
|
|
delete(client.pushSubscriptions, endpoint)
|
|
|
|
client.rebuildPushSubscriptionCache()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) recordPush(endpoint string, success bool) {
|
|
|
|
now := time.Now().UTC()
|
|
|
|
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
subscription, ok := client.pushSubscriptions[endpoint]
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if success {
|
|
|
|
subscription.LastSuccess = now
|
|
|
|
}
|
|
|
|
// TODO we may want to track failures in some way in the future
|
|
|
|
}
|
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
func (channel *Channel) Name() string {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.name
|
|
|
|
}
|
|
|
|
|
2017-10-30 10:21:47 +01:00
|
|
|
func (channel *Channel) NameCasefolded() string {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.nameCasefolded
|
|
|
|
}
|
|
|
|
|
2019-03-12 00:24:45 +01:00
|
|
|
func (channel *Channel) Rename(name, nameCasefolded string) {
|
2017-10-30 10:21:47 +01:00
|
|
|
channel.stateMutex.Lock()
|
2019-03-12 00:24:45 +01:00
|
|
|
channel.name = name
|
2020-08-04 16:13:29 +02:00
|
|
|
if channel.nameCasefolded != nameCasefolded {
|
|
|
|
channel.nameCasefolded = nameCasefolded
|
|
|
|
if channel.registeredFounder != "" {
|
|
|
|
channel.registeredTime = time.Now().UTC()
|
|
|
|
}
|
2019-03-12 00:24:45 +01:00
|
|
|
}
|
|
|
|
channel.stateMutex.Unlock()
|
2017-10-30 10:21:47 +01:00
|
|
|
}
|
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
func (channel *Channel) Members() (result []*Client) {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.membersCache
|
|
|
|
}
|
|
|
|
|
2018-12-28 19:45:55 +01:00
|
|
|
func (channel *Channel) setUserLimit(limit int) {
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.stateMutex.Lock()
|
|
|
|
channel.userLimit = limit
|
|
|
|
channel.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (channel *Channel) setKey(key string) {
|
|
|
|
channel.stateMutex.Lock()
|
2018-04-04 03:49:40 +02:00
|
|
|
defer channel.stateMutex.Unlock()
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.key = key
|
|
|
|
}
|
|
|
|
|
2017-11-09 04:19:50 +01:00
|
|
|
func (channel *Channel) Founder() string {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.registeredFounder
|
|
|
|
}
|
2019-03-12 00:24:45 +01:00
|
|
|
|
2019-04-23 06:05:12 +02:00
|
|
|
func (channel *Channel) HighestUserMode(client *Client) (result modes.Mode) {
|
|
|
|
channel.stateMutex.RLock()
|
2024-05-06 08:30:00 +02:00
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
if clientData, ok := channel.members[client]; ok {
|
|
|
|
return clientData.modes.HighestChannelUserMode()
|
|
|
|
}
|
|
|
|
return
|
2019-04-23 06:05:12 +02:00
|
|
|
}
|
2020-02-19 01:38:42 +01:00
|
|
|
|
|
|
|
func (channel *Channel) Settings() (result ChannelSettings) {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
result = channel.settings
|
|
|
|
channel.stateMutex.RUnlock()
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (channel *Channel) SetSettings(settings ChannelSettings) {
|
|
|
|
channel.stateMutex.Lock()
|
|
|
|
channel.settings = settings
|
|
|
|
channel.stateMutex.Unlock()
|
|
|
|
channel.MarkDirty(IncludeSettings)
|
|
|
|
}
|
2020-12-14 11:00:21 +01:00
|
|
|
|
|
|
|
func (channel *Channel) setForward(forward string) {
|
|
|
|
channel.stateMutex.Lock()
|
|
|
|
channel.forward = forward
|
|
|
|
channel.stateMutex.Unlock()
|
|
|
|
}
|
2020-12-15 10:00:44 +01:00
|
|
|
|
|
|
|
func (channel *Channel) Ctime() (ctime time.Time) {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
ctime = channel.createdTime
|
|
|
|
channel.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
2021-01-19 14:49:45 +01:00
|
|
|
|
|
|
|
func (channel *Channel) getAmode(cfaccount string) (result modes.Mode) {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.accountToUMode[cfaccount]
|
|
|
|
}
|
2023-01-04 11:06:21 +01:00
|
|
|
|
|
|
|
func (channel *Channel) UUID() utils.UUID {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.uuid
|
|
|
|
}
|