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 (
|
2019-05-08 10:11:54 +02:00
|
|
|
"net"
|
2019-05-09 20:18:30 +02:00
|
|
|
"sync/atomic"
|
2019-03-12 00:24:45 +01:00
|
|
|
"time"
|
2019-05-09 20:18:30 +02:00
|
|
|
"unsafe"
|
2019-03-12 00:24:45 +01:00
|
|
|
|
2019-02-19 08:54:57 +01:00
|
|
|
"github.com/oragono/oragono/irc/languages"
|
2018-02-03 11:21:32 +01:00
|
|
|
"github.com/oragono/oragono/irc/modes"
|
|
|
|
)
|
2017-10-05 15:39:57 +02:00
|
|
|
|
2019-02-19 08:54:57 +01:00
|
|
|
func (server *Server) Config() (config *Config) {
|
2019-05-09 20:18:30 +02:00
|
|
|
return (*Config)(atomic.LoadPointer(&server.config))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (server *Server) SetConfig(config *Config) {
|
|
|
|
atomic.StorePointer(&server.config, unsafe.Pointer(config))
|
2018-03-18 02:32:12 +01:00
|
|
|
}
|
|
|
|
|
2017-11-09 04:19:50 +01:00
|
|
|
func (server *Server) ChannelRegistrationEnabled() bool {
|
2018-07-16 09:46:40 +02:00
|
|
|
return server.Config().Channels.Registration.Enabled
|
2017-11-09 04:19:50 +01:00
|
|
|
}
|
|
|
|
|
2018-02-11 11:30:40 +01:00
|
|
|
func (server *Server) AccountConfig() *AccountConfig {
|
2018-07-16 09:46:40 +02:00
|
|
|
return &server.Config().Accounts
|
2018-03-22 16:04:21 +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
|
|
|
|
}
|
|
|
|
|
2019-04-12 06:08:46 +02:00
|
|
|
func (client *Client) Sessions() (sessions []*Session) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
sessions = make([]*Session, len(client.sessions))
|
|
|
|
copy(sessions, client.sessions)
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-05-22 03:40:25 +02:00
|
|
|
func (client *Client) GetSessionByResumeID(resumeID string) (result *Session) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
|
|
|
|
for _, session := range client.sessions {
|
|
|
|
if session.resumeID == resumeID {
|
|
|
|
return session
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-05-08 10:11:54 +02:00
|
|
|
type SessionData struct {
|
|
|
|
ctime time.Time
|
|
|
|
atime time.Time
|
|
|
|
ip net.IP
|
|
|
|
hostname string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) AllSessionData(currentSession *Session) (data []SessionData, currentIndex int) {
|
|
|
|
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{
|
|
|
|
atime: session.atime,
|
|
|
|
ctime: session.ctime,
|
|
|
|
hostname: session.rawHostname,
|
|
|
|
}
|
|
|
|
if session.proxiedIP != nil {
|
|
|
|
data[i].ip = session.proxiedIP
|
|
|
|
} else {
|
|
|
|
data[i].ip = session.realIP
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-04-12 06:08:46 +02:00
|
|
|
func (client *Client) AddSession(session *Session) (success bool) {
|
|
|
|
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
|
|
|
|
switch client.brbTimer.state {
|
|
|
|
case BrbDisabled:
|
|
|
|
if len(client.sessions) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
case BrbDead:
|
2019-04-12 06:08:46 +02:00
|
|
|
return false
|
2019-05-22 03:40:25 +02:00
|
|
|
// default: BrbEnabled or BrbSticky, proceed
|
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
|
|
|
|
client.sessions = append(client.sessions, session)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-05-22 03:40:25 +02:00
|
|
|
func (session *Session) SetResumeID(resumeID string) {
|
|
|
|
session.client.stateMutex.Lock()
|
|
|
|
session.resumeID = resumeID
|
|
|
|
session.client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-04-28 21:10:03 +02:00
|
|
|
func (client *Client) Away() (result bool) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
result = client.away
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetAway(away bool, awayMessage string) (changed bool) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
changed = away != client.away
|
|
|
|
client.away = away
|
|
|
|
client.awayMessage = awayMessage
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-02-12 06:27:57 +01:00
|
|
|
func (client *Client) ResumeID() string {
|
2018-11-26 11:23:27 +01:00
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
2019-02-12 06:27:57 +01:00
|
|
|
return client.resumeID
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetResumeID(id string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
client.resumeID = id
|
2018-11-26 11:23:27 +01:00
|
|
|
}
|
|
|
|
|
2018-04-19 08:48:19 +02:00
|
|
|
func (client *Client) Oper() *Oper {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.oper
|
|
|
|
}
|
|
|
|
|
2017-10-15 18:24:28 +02:00
|
|
|
func (client *Client) Registered() bool {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
defer client.stateMutex.RUnlock()
|
|
|
|
return client.registered
|
|
|
|
}
|
|
|
|
|
2019-02-10 02:01:47 +01:00
|
|
|
func (client *Client) SetRegistered() {
|
|
|
|
// `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
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.registered = true
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
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-05-22 03:40:25 +02:00
|
|
|
func (client *Client) SetRawHostname(rawHostname string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
|
|
|
|
|
|
|
client.rawHostname = rawHostname
|
|
|
|
client.updateNickMaskNoMutex()
|
|
|
|
}
|
|
|
|
|
2019-02-17 20:29:04 +01:00
|
|
|
func (client *Client) AwayMessage() (result string) {
|
|
|
|
client.stateMutex.RLock()
|
|
|
|
result = client.awayMessage
|
|
|
|
client.stateMutex.RUnlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetAwayMessage(message string) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.awayMessage = message
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2018-02-20 10:50:46 +01:00
|
|
|
func (client *Client) SetAccountName(account string) (changed bool) {
|
2018-02-11 11:30:40 +01:00
|
|
|
var casefoldedAccount string
|
2018-02-20 10:50:46 +01:00
|
|
|
var err error
|
2018-02-11 11:30:40 +01:00
|
|
|
if account != "" {
|
2018-02-20 10:50:46 +01:00
|
|
|
if casefoldedAccount, err = CasefoldName(account); err != nil {
|
|
|
|
return
|
|
|
|
}
|
2018-02-11 11:30:40 +01:00
|
|
|
}
|
2018-02-20 10:50:46 +01:00
|
|
|
|
2018-02-11 11:30:40 +01:00
|
|
|
client.stateMutex.Lock()
|
|
|
|
defer client.stateMutex.Unlock()
|
2018-02-20 10:50:46 +01:00
|
|
|
changed = client.account != casefoldedAccount
|
2018-02-11 11:30:40 +01:00
|
|
|
client.account = casefoldedAccount
|
|
|
|
client.accountName = account
|
2018-02-20 10:50:46 +01:00
|
|
|
return
|
2017-11-09 04:19:50 +01:00
|
|
|
}
|
|
|
|
|
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) {
|
|
|
|
client.stateMutex.Lock()
|
|
|
|
client.accountSettings = settings
|
|
|
|
client.stateMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
return client.flags.HasMode(mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (client *Client) SetMode(mode modes.Mode, on bool) bool {
|
|
|
|
return client.flags.SetMode(mode, on)
|
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
|
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
|
|
|
|
}
|
|
|
|
|
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
|
2017-10-30 10:21:47 +01:00
|
|
|
channel.nameCasefolded = nameCasefolded
|
2019-03-12 00:24:45 +01:00
|
|
|
if channel.registeredFounder != "" {
|
2019-05-12 09:12:50 +02:00
|
|
|
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) Key() string {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
defer channel.stateMutex.RUnlock()
|
|
|
|
return channel.key
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
func (channel *Channel) DirtyBits() (dirtyBits uint) {
|
|
|
|
channel.stateMutex.Lock()
|
|
|
|
dirtyBits = channel.dirtyBits
|
|
|
|
channel.stateMutex.Unlock()
|
|
|
|
return
|
|
|
|
}
|
2019-04-23 06:05:12 +02:00
|
|
|
|
|
|
|
func (channel *Channel) HighestUserMode(client *Client) (result modes.Mode) {
|
|
|
|
channel.stateMutex.RLock()
|
|
|
|
clientModes := channel.members[client]
|
|
|
|
channel.stateMutex.RUnlock()
|
|
|
|
return clientModes.HighestChannelUserMode()
|
|
|
|
}
|