2016-06-15 13:50:56 +02:00
// Copyright (c) 2012-2014 Jeremy Latt
// Copyright (c) 2014-2015 Edmund Huber
2017-03-27 14:15:02 +02:00
// Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
2016-06-15 13:50:56 +02:00
// released under the MIT license
2012-04-07 20:44:59 +02:00
package irc
import (
2014-02-12 01:35:32 +01:00
"bufio"
2016-04-13 12:45:09 +02:00
"crypto/tls"
2016-09-05 10:45:42 +02:00
"encoding/base64"
2014-02-09 02:10:04 +01:00
"fmt"
2012-04-07 20:44:59 +02:00
"log"
2017-01-14 10:52:47 +01:00
"math/rand"
2012-04-07 20:44:59 +02:00
"net"
2018-03-13 19:46:39 +01:00
"net/http"
_ "net/http/pprof"
2014-02-12 01:35:32 +01:00
"os"
2014-02-25 20:11:34 +01:00
"os/signal"
2016-06-17 14:17:42 +02:00
"strconv"
2014-02-18 03:10:52 +01:00
"strings"
2016-10-22 12:54:04 +02:00
"sync"
2014-03-02 20:36:00 +01:00
"syscall"
2012-04-18 07:21:41 +02:00
"time"
2014-03-13 01:52:25 +01:00
2017-06-15 18:14:19 +02:00
"github.com/goshuirc/irc-go/ircfmt"
"github.com/goshuirc/irc-go/ircmsg"
2017-09-29 04:07:52 +02:00
"github.com/oragono/oragono/irc/caps"
2017-10-09 23:37:13 +02:00
"github.com/oragono/oragono/irc/connection_limits"
2017-10-05 15:39:57 +02:00
"github.com/oragono/oragono/irc/isupport"
2018-02-03 10:46:14 +01:00
"github.com/oragono/oragono/irc/languages"
2017-06-14 20:00:53 +02:00
"github.com/oragono/oragono/irc/logger"
2018-02-03 11:21:32 +01:00
"github.com/oragono/oragono/irc/modes"
2017-10-05 16:03:53 +02:00
"github.com/oragono/oragono/irc/passwd"
2017-06-14 20:00:53 +02:00
"github.com/oragono/oragono/irc/sno"
2017-10-05 15:47:43 +02:00
"github.com/oragono/oragono/irc/utils"
2016-08-19 15:21:52 +02:00
"github.com/tidwall/buntdb"
2016-06-17 14:17:42 +02:00
)
2014-03-08 23:20:36 +01:00
2016-10-23 15:05:00 +02:00
var (
2017-09-28 23:18:08 +02:00
// common error line to sub values into
errorMsg , _ = ( & [ ] ircmsg . IrcMessage { ircmsg . MakeMessage ( nil , "" , "ERROR" , "%s " ) } [ 0 ] ) . Line ( )
2017-07-26 07:19:40 +02:00
// common error responses
2017-09-28 23:18:08 +02:00
couldNotParseIPMsg , _ = ( & [ ] ircmsg . IrcMessage { ircmsg . MakeMessage ( nil , "" , "ERROR" , "Unable to parse your IP address" ) } [ 0 ] ) . Line ( )
2017-10-30 10:21:47 +01:00
2018-02-03 11:21:32 +01:00
// supportedUserModesString acts as a cache for when we introduce users
supportedUserModesString = modes . SupportedUserModes . String ( )
// supportedChannelModesString acts as a cache for when we introduce users
supportedChannelModesString = modes . SupportedChannelModes . String ( )
2018-02-03 12:15:07 +01:00
// SupportedCapabilities are the caps we advertise.
// MaxLine, SASL and STS are set during server startup.
2018-02-04 12:32:48 +01:00
SupportedCapabilities = caps . NewSet ( caps . AccountTag , caps . AccountNotify , caps . AwayNotify , caps . Batch , caps . CapNotify , caps . ChgHost , caps . EchoMessage , caps . ExtendedJoin , caps . InviteNotify , caps . LabeledResponse , caps . Languages , caps . MessageTags , caps . MultiPrefix , caps . Rename , caps . Resume , caps . ServerTime , caps . UserhostInNames )
2018-02-03 12:15:07 +01:00
// CapValues are the actual values we advertise to v3.2 clients.
// actual values are set during server startup.
CapValues = caps . NewValues ( )
2017-09-28 07:30:53 +02:00
)
2016-11-06 04:47:13 +01:00
2017-01-13 15:22:42 +01:00
// Limits holds the maximum limits for various things such as topic lengths.
2016-09-12 04:22:50 +02:00
type Limits struct {
2016-10-16 12:14:56 +02:00
AwayLen int
ChannelLen int
KickLen int
MonitorEntries int
NickLen int
TopicLen int
2016-10-23 16:50:18 +02:00
ChanListModes int
2017-01-13 15:22:42 +01:00
LineLen LineLenLimits
}
// LineLenLimits holds the maximum limits for IRC lines.
type LineLenLimits struct {
Tags int
Rest int
2016-09-12 04:22:50 +02:00
}
2017-09-12 00:40:15 +02:00
// ListenerWrapper wraps a listener so it can be safely reconfigured or stopped
type ListenerWrapper struct {
2017-09-25 03:29:27 +02:00
listener net . Listener
tlsConfig * tls . Config
shouldStop bool
2017-09-12 00:40:15 +02:00
// protects atomic update of tlsConfig and shouldStop:
2017-11-22 10:41:11 +01:00
configMutex sync . Mutex // tier 1
2016-10-22 12:54:04 +02:00
}
// Server is the main Oragono server.
2012-04-07 20:44:59 +02:00
type Server struct {
2018-02-11 11:30:40 +01:00
accounts * AccountManager
batches * BatchManager
channelRegistrationEnabled bool
channels * ChannelManager
channelRegistry * ChannelRegistry
checkIdent bool
clients * ClientManager
2018-03-22 16:04:21 +01:00
config * Config
2018-02-11 11:30:40 +01:00
configFilename string
configurableStateMutex sync . RWMutex // tier 1; generic protection for server state modified by rehash()
connectionLimiter * connection_limits . Limiter
connectionThrottler * connection_limits . Throttler
ctime time . Time
defaultChannelModes modes . Modes
dlines * DLineManager
loggingRawIO bool
isupport * isupport . List
klines * KLineManager
languages * languages . Manager
limits Limits
listeners map [ string ] * ListenerWrapper
logger * logger . Manager
2018-03-19 05:24:20 +01:00
maxSendQBytes uint32
2018-02-11 11:30:40 +01:00
monitorManager * MonitorManager
motdLines [ ] string
name string
nameCasefolded string
networkName string
operators map [ string ] Oper
operclasses map [ string ] OperClass
password [ ] byte
passwords * passwd . SaltedManager
recoverFromErrors bool
rehashMutex sync . Mutex // tier 4
rehashSignal chan os . Signal
2018-03-13 19:46:39 +01:00
pprofServer * http . Server
2018-02-11 11:30:40 +01:00
proxyAllowedFrom [ ] string
signals chan os . Signal
snomasks * SnoManager
store * buntdb . DB
2018-02-19 03:52:39 +01:00
storeFilename string
2018-02-11 11:30:40 +01:00
stsEnabled bool
webirc [ ] webircConfig
whoWas * WhoWasList
2018-04-20 22:48:15 +02:00
stats * Stats
2012-04-18 03:16:57 +02:00
}
2014-03-13 01:52:25 +01:00
var (
2016-10-23 03:48:57 +02:00
// ServerExitSignals are the signals the server will exit on.
ServerExitSignals = [ ] os . Signal {
2016-08-14 06:13:01 +02:00
syscall . SIGINT ,
syscall . SIGTERM ,
syscall . SIGQUIT ,
}
2014-03-13 01:52:25 +01:00
)
2016-06-28 17:09:07 +02:00
type clientConn struct {
Conn net . Conn
IsTLS bool
}
2016-10-19 13:38:31 +02:00
// NewServer returns a new Oragono server.
2017-09-28 07:30:53 +02:00
func NewServer ( config * Config , logger * logger . Manager ) ( * Server , error ) {
// initialize data structures
2012-12-09 21:51:50 +01:00
server := & Server {
2018-01-27 15:30:30 +01:00
batches : NewBatchManager ( ) ,
2017-10-30 10:21:47 +01:00
channels : NewChannelManager ( ) ,
2017-11-22 10:41:11 +01:00
clients : NewClientManager ( ) ,
2017-10-09 23:37:13 +02:00
connectionLimiter : connection_limits . NewLimiter ( ) ,
connectionThrottler : connection_limits . NewThrottler ( ) ,
2018-02-03 10:46:14 +01:00
languages : languages . NewManager ( config . Languages . Default , config . Languages . Data ) ,
2017-10-09 23:37:13 +02:00
listeners : make ( map [ string ] * ListenerWrapper ) ,
logger : logger ,
monitorManager : NewMonitorManager ( ) ,
rehashSignal : make ( chan os . Signal , 1 ) ,
signals : make ( chan os . Signal , len ( ServerExitSignals ) ) ,
snomasks : NewSnoManager ( ) ,
whoWas : NewWhoWasList ( config . Limits . WhowasEntries ) ,
2018-04-20 22:48:15 +02:00
stats : NewStats ( ) ,
2012-12-15 23:34:20 +01:00
}
2014-02-09 19:07:40 +01:00
2017-09-28 07:30:53 +02:00
if err := server . applyConfig ( config , true ) ; err != nil {
return nil , err
2016-08-19 15:21:52 +02:00
}
2016-09-04 11:25:33 +02:00
2018-01-23 12:17:14 +01:00
// generate help info
if err := GenerateHelpIndices ( server . languages ) ; err != nil {
return nil , err
}
2018-04-01 03:54:49 +02:00
// confirm help entries for ChanServ exist.
// this forces people to write help entries for every single CS command.
for commandName , commandInfo := range chanservCommands {
if commandInfo . help == "" || commandInfo . helpShort == "" {
return nil , fmt . Errorf ( "Help entry does not exist for ChanServ command %s" , commandName )
}
}
2018-04-01 03:22:06 +02:00
// confirm help entries for NickServ exist.
// this forces people to write help entries for every single NS command.
for commandName , commandInfo := range nickservCommands {
if commandInfo . help == "" || commandInfo . helpShort == "" {
return nil , fmt . Errorf ( "Help entry does not exist for NickServ command %s" , commandName )
}
}
2016-08-14 06:13:01 +02:00
// Attempt to clean up when receiving these signals.
2016-10-23 03:48:57 +02:00
signal . Notify ( server . signals , ServerExitSignals ... )
2016-10-22 13:20:08 +02:00
signal . Notify ( server . rehashSignal , syscall . SIGHUP )
2014-03-02 20:51:29 +01:00
2017-03-06 06:50:23 +01:00
return server , nil
2016-10-19 13:38:31 +02:00
}
// setISupport sets up our RPL_ISUPPORT reply.
func ( server * Server ) setISupport ( ) {
2016-10-26 16:51:55 +02:00
maxTargetsString := strconv . Itoa ( maxTargets )
2017-10-02 10:42:50 +02:00
server . configurableStateMutex . RLock ( )
2016-04-12 07:38:42 +02:00
// add RPL_ISUPPORT tokens
2017-10-05 15:39:57 +02:00
isupport := isupport . NewList ( )
2017-10-02 10:42:50 +02:00
isupport . Add ( "AWAYLEN" , strconv . Itoa ( server . limits . AwayLen ) )
2017-12-26 03:30:04 +01:00
isupport . Add ( "CASEMAPPING" , "ascii" )
2018-02-03 11:21:32 +01:00
isupport . Add ( "CHANMODES" , strings . Join ( [ ] string { modes . Modes { modes . BanMask , modes . ExceptMask , modes . InviteMask } . String ( ) , "" , modes . Modes { modes . UserLimit , modes . Key } . String ( ) , modes . Modes { modes . InviteOnly , modes . Moderated , modes . NoOutside , modes . OpOnlyTopic , modes . ChanRoleplaying , modes . Secret } . String ( ) } , "," ) )
2017-10-02 10:42:50 +02:00
isupport . Add ( "CHANNELLEN" , strconv . Itoa ( server . limits . ChannelLen ) )
isupport . Add ( "CHANTYPES" , "#" )
isupport . Add ( "ELIST" , "U" )
isupport . Add ( "EXCEPTS" , "" )
isupport . Add ( "INVEX" , "" )
isupport . Add ( "KICKLEN" , strconv . Itoa ( server . limits . KickLen ) )
isupport . Add ( "MAXLIST" , fmt . Sprintf ( "beI:%s" , strconv . Itoa ( server . limits . ChanListModes ) ) )
isupport . Add ( "MAXTARGETS" , maxTargetsString )
isupport . Add ( "MODES" , "" )
isupport . Add ( "MONITOR" , strconv . Itoa ( server . limits . MonitorEntries ) )
isupport . Add ( "NETWORK" , server . networkName )
isupport . Add ( "NICKLEN" , strconv . Itoa ( server . limits . NickLen ) )
isupport . Add ( "PREFIX" , "(qaohv)~&@%+" )
isupport . Add ( "RPCHAN" , "E" )
isupport . Add ( "RPUSER" , "E" )
isupport . Add ( "STATUSMSG" , "~&@%+" )
isupport . Add ( "TARGMAX" , fmt . Sprintf ( "NAMES:1,LIST:1,KICK:1,WHOIS:1,USERHOST:10,PRIVMSG:%s,TAGMSG:%s,NOTICE:%s,MONITOR:" , maxTargetsString , maxTargetsString , maxTargetsString ) )
isupport . Add ( "TOPICLEN" , strconv . Itoa ( server . limits . TopicLen ) )
2017-12-26 03:30:04 +01:00
isupport . Add ( "UTF8MAPPING" , casemappingName )
2016-09-04 11:25:33 +02:00
// account registration
2018-03-22 16:04:21 +01:00
if server . config . Accounts . Registration . Enabled {
2016-09-04 11:25:33 +02:00
// 'none' isn't shown in the REGCALLBACKS vars
2016-09-04 13:15:28 +02:00
var enabledCallbacks [ ] string
2018-03-22 16:04:21 +01:00
for _ , name := range server . config . Accounts . Registration . EnabledCallbacks {
2016-09-04 12:08:53 +02:00
if name != "*" {
2016-09-04 13:15:28 +02:00
enabledCallbacks = append ( enabledCallbacks , name )
2016-09-04 11:25:33 +02:00
}
}
2018-04-01 04:04:25 +02:00
isupport . Add ( "ACCCOMMANDS" , "CREATE,VERIFY" )
2017-10-02 10:42:50 +02:00
isupport . Add ( "REGCALLBACKS" , strings . Join ( enabledCallbacks , "," ) )
isupport . Add ( "REGCREDTYPES" , "passphrase,certfp" )
2016-09-04 11:25:33 +02:00
}
2017-10-02 10:42:50 +02:00
server . configurableStateMutex . RUnlock ( )
isupport . RegenerateCachedReply ( )
server . configurableStateMutex . Lock ( )
server . isupport = isupport
server . configurableStateMutex . Unlock ( )
2014-02-26 00:57:35 +01:00
}
2018-02-03 11:21:32 +01:00
func loadChannelList ( channel * Channel , list string , maskMode modes . Mode ) {
2014-03-08 03:14:02 +01:00
if list == "" {
return
}
2016-10-11 15:51:46 +02:00
channel . lists [ maskMode ] . AddAll ( strings . Split ( list , " " ) )
2014-03-08 03:14:02 +01:00
}
2017-03-06 00:27:08 +01:00
// Shutdown shuts down the server.
2014-03-02 20:36:00 +01:00
func ( server * Server ) Shutdown ( ) {
2016-09-05 11:39:16 +02:00
//TODO(dan): Make sure we disallow new nicks
2017-11-22 10:41:11 +01:00
for _ , client := range server . clients . AllClients ( ) {
2016-06-19 13:59:18 +02:00
client . Notice ( "Server is shutting down" )
2014-03-02 20:36:00 +01:00
}
2016-08-14 06:13:01 +02:00
2016-09-05 11:39:16 +02:00
if err := server . store . Close ( ) ; err != nil {
2017-03-10 13:02:08 +01:00
server . logger . Error ( "shutdown" , fmt . Sprintln ( "Could not close datastore:" , err ) )
2016-08-14 06:13:01 +02:00
}
2014-03-02 20:36:00 +01:00
}
2016-10-23 03:48:57 +02:00
// Run starts the server.
2014-02-24 04:13:45 +01:00
func ( server * Server ) Run ( ) {
2016-09-05 14:14:20 +02:00
// defer closing db/store
defer server . store . Close ( )
2017-12-11 03:18:16 +01:00
for {
2014-02-17 07:38:43 +01:00
select {
2014-02-25 20:11:34 +01:00
case <- server . signals :
2014-03-02 20:36:00 +01:00
server . Shutdown ( )
2017-12-11 03:18:16 +01:00
return
2014-02-25 20:11:34 +01:00
2016-10-22 13:20:08 +02:00
case <- server . rehashSignal :
2017-09-08 12:02:54 +02:00
go func ( ) {
2017-12-11 03:18:16 +01:00
server . logger . Info ( "rehash" , "Rehashing due to SIGHUP" )
2017-09-08 12:02:54 +02:00
err := server . rehash ( )
if err != nil {
server . logger . Error ( "rehash" , fmt . Sprintln ( "Failed to rehash:" , err . Error ( ) ) )
}
} ( )
2017-12-11 03:18:16 +01:00
}
}
}
2016-10-22 13:20:08 +02:00
2017-12-11 03:18:16 +01:00
func ( server * Server ) acceptClient ( conn clientConn ) {
// check IP address
2018-02-01 21:53:49 +01:00
ipaddr := utils . AddrToIP ( conn . Conn . RemoteAddr ( ) )
if ipaddr != nil {
isBanned , banMsg := server . checkBans ( ipaddr )
if isBanned {
// this might not show up properly on some clients, but our objective here is just to close the connection out before it has a load impact on us
conn . Conn . Write ( [ ] byte ( fmt . Sprintf ( errorMsg , banMsg ) ) )
conn . Conn . Close ( )
return
}
2017-12-11 03:18:16 +01:00
}
2017-07-26 07:19:40 +02:00
2017-12-11 03:18:16 +01:00
server . logger . Debug ( "localconnect-ip" , fmt . Sprintf ( "Client connecting from %v" , ipaddr ) )
// prolly don't need to alert snomasks on this, only on connection reg
2017-07-26 07:19:40 +02:00
2017-12-11 03:18:16 +01:00
NewClient ( server , conn . Conn , conn . IsTLS )
2014-02-14 03:39:33 +01:00
}
2017-09-28 23:18:08 +02:00
func ( server * Server ) checkBans ( ipaddr net . IP ) ( banned bool , message string ) {
2017-09-28 20:19:39 +02:00
// check DLINEs
isBanned , info := server . dlines . CheckIP ( ipaddr )
if isBanned {
2017-10-09 19:17:49 +02:00
server . logger . Info ( "localconnect-ip" , fmt . Sprintf ( "Client from %v rejected by d-line" , ipaddr ) )
2017-09-28 23:18:08 +02:00
return true , info . BanMessage ( "You are banned from this server (%s)" )
2017-09-28 20:19:39 +02:00
}
// check connection limits
2017-10-09 23:37:13 +02:00
err := server . connectionLimiter . AddClient ( ipaddr , false )
2017-09-28 20:19:39 +02:00
if err != nil {
// too many connections from one client, tell the client and close the connection
2017-10-09 19:17:49 +02:00
server . logger . Info ( "localconnect-ip" , fmt . Sprintf ( "Client from %v rejected for connection limit" , ipaddr ) )
2017-09-28 23:18:08 +02:00
return true , "Too many clients from your network"
2017-09-28 20:19:39 +02:00
}
// check connection throttle
2017-10-09 23:37:13 +02:00
err = server . connectionThrottler . AddClient ( ipaddr )
2017-09-28 20:19:39 +02:00
if err != nil {
// too many connections too quickly from client, tell them and close the connection
2017-10-09 23:37:13 +02:00
duration := server . connectionThrottler . BanDuration ( )
2017-09-28 20:19:39 +02:00
length := & IPRestrictTime {
2017-10-09 07:47:04 +02:00
Duration : duration ,
Expires : time . Now ( ) . Add ( duration ) ,
2017-09-28 20:19:39 +02:00
}
2017-11-19 01:27:40 +01:00
server . dlines . AddIP ( ipaddr , length , server . connectionThrottler . BanMessage ( ) , "Exceeded automated connection throttle" , "auto.connection.throttler" )
2017-09-28 20:19:39 +02:00
// they're DLINE'd for 15 minutes or whatever, so we can reset the connection throttle now,
// and once their temporary DLINE is finished they can fill up the throttler again
2017-10-09 23:37:13 +02:00
server . connectionThrottler . ResetFor ( ipaddr )
2017-09-28 20:19:39 +02:00
// this might not show up properly on some clients, but our objective here is just to close it out before it has a load impact on us
2017-10-09 19:17:49 +02:00
server . logger . Info (
"localconnect-ip" ,
fmt . Sprintf ( "Client from %v exceeded connection throttle, d-lining for %v" , ipaddr , duration ) )
2017-10-09 23:37:13 +02:00
return true , server . connectionThrottler . BanMessage ( )
2017-09-28 20:19:39 +02:00
}
2017-09-28 23:18:08 +02:00
return false , ""
2017-09-28 20:19:39 +02:00
}
2014-02-20 04:30:49 +01:00
//
2016-10-22 12:54:04 +02:00
// IRC protocol listeners
2014-02-20 04:30:49 +01:00
//
2016-10-23 03:48:57 +02:00
// createListener starts the given listeners.
2017-09-12 00:40:15 +02:00
func ( server * Server ) createListener ( addr string , tlsConfig * tls . Config ) * ListenerWrapper {
2016-10-22 12:54:04 +02:00
// make listener
2018-02-01 21:53:49 +01:00
var listener net . Listener
var err error
2018-02-11 11:30:40 +01:00
addr = strings . TrimPrefix ( addr , "unix:" )
if strings . HasPrefix ( addr , "/" ) {
2018-02-01 21:53:49 +01:00
// https://stackoverflow.com/a/34881585
os . Remove ( addr )
listener , err = net . Listen ( "unix" , addr )
} else {
listener , err = net . Listen ( "tcp" , addr )
}
2012-04-07 20:44:59 +02:00
if err != nil {
2016-10-23 03:48:57 +02:00
log . Fatal ( server , "listen error: " , err )
2012-04-07 20:44:59 +02:00
}
2016-10-22 12:54:04 +02:00
// throw our details to the server so we can be modified/killed later
2017-09-12 00:40:15 +02:00
wrapper := ListenerWrapper {
listener : listener ,
tlsConfig : tlsConfig ,
shouldStop : false ,
2016-10-22 12:54:04 +02:00
}
2017-09-12 00:40:15 +02:00
var shouldStop bool
2016-10-22 12:54:04 +02:00
// setup accept goroutine
2014-03-14 01:19:39 +01:00
go func ( ) {
for {
conn , err := listener . Accept ( )
2014-02-17 07:30:01 +01:00
2017-09-12 00:40:15 +02:00
// synchronously access config data:
// whether TLS is enabled and whether we should stop listening
wrapper . configMutex . Lock ( )
shouldStop = wrapper . shouldStop
tlsConfig = wrapper . tlsConfig
wrapper . configMutex . Unlock ( )
2016-10-22 12:54:04 +02:00
if err == nil {
2017-09-12 00:40:15 +02:00
if tlsConfig != nil {
conn = tls . Server ( conn , tlsConfig )
}
2016-10-22 12:54:04 +02:00
newConn := clientConn {
Conn : conn ,
2017-09-12 00:40:15 +02:00
IsTLS : tlsConfig != nil ,
2016-10-22 12:54:04 +02:00
}
2017-09-12 00:40:15 +02:00
// hand off the connection
2017-12-11 03:18:16 +01:00
go server . acceptClient ( newConn )
2016-06-28 17:09:07 +02:00
}
2017-09-12 00:40:15 +02:00
if shouldStop {
listener . Close ( )
return
2016-10-22 12:54:04 +02:00
}
2014-03-14 01:19:39 +01:00
}
} ( )
2017-09-08 02:20:08 +02:00
2017-09-12 00:40:15 +02:00
return & wrapper
2012-12-09 07:54:58 +01:00
}
2017-01-14 10:52:47 +01:00
// generateMessageID returns a network-unique message ID.
func ( server * Server ) generateMessageID ( ) string {
2018-01-03 15:13:32 +01:00
// we don't need the full like 30 chars since the unixnano below handles
// most of our uniqueness requirements, so just truncate at 5
2018-01-03 15:21:35 +01:00
lastbit := strconv . FormatInt ( rand . Int63 ( ) , 36 )
2018-01-03 15:13:32 +01:00
if 5 < len ( lastbit ) {
2018-01-03 15:21:35 +01:00
lastbit = lastbit [ : 4 ]
2018-01-03 15:13:32 +01:00
}
2018-01-03 15:21:35 +01:00
return fmt . Sprintf ( "%s%s" , strconv . FormatInt ( time . Now ( ) . UTC ( ) . UnixNano ( ) , 36 ) , lastbit )
2017-01-14 10:52:47 +01:00
}
2014-02-15 03:28:36 +01:00
//
2012-12-09 07:54:58 +01:00
// server functionality
2014-02-15 03:28:36 +01:00
//
2012-12-09 07:54:58 +01:00
2016-10-23 03:48:57 +02:00
func ( server * Server ) tryRegister ( c * Client ) {
2018-02-27 03:44:03 +01:00
if c . Registered ( ) {
return
}
preregNick := c . PreregNick ( )
if preregNick == "" || ! c . HasUsername ( ) || c . capState == caps . NegotiatingState {
return
}
// client MUST send PASS (or AUTHENTICATE, if skip-server-password is set)
// before completing the other registration commands
if ! c . Authorized ( ) {
c . Quit ( c . t ( "Bad password" ) )
c . destroy ( false )
return
}
rb := NewResponseBuffer ( c )
nickAssigned := performNickChange ( server , c , c , preregNick , rb )
rb . Send ( )
if ! nickAssigned {
2018-02-27 11:22:01 +01:00
c . SetPreregNick ( "" )
2014-03-13 01:52:25 +01:00
return
2012-12-15 23:34:20 +01:00
}
2017-01-11 13:38:16 +01:00
2018-04-24 13:47:35 +02:00
// Check if connection requires SASL
if connectionRequiresSasl ( c ) {
c . destroy ( false )
return
}
2017-01-11 13:38:16 +01:00
// check KLINEs
isBanned , info := server . klines . CheckMasks ( c . AllNickmasks ( ) ... )
if isBanned {
reason := info . Reason
if info . Time != nil {
reason += fmt . Sprintf ( " [%s]" , info . Time . Duration . String ( ) )
}
2018-01-22 12:26:01 +01:00
c . Quit ( fmt . Sprintf ( c . t ( "You are banned from this server (%s)" ) , reason ) )
2018-01-21 02:59:52 +01:00
c . destroy ( false )
2017-01-11 13:38:16 +01:00
return
}
2018-04-20 22:48:15 +02:00
// count new user in statistics
server . stats . ChangeTotal ( 1 )
2017-01-11 13:38:16 +01:00
// continue registration
2017-05-01 11:03:04 +02:00
server . logger . Debug ( "localconnect" , fmt . Sprintf ( "Client registered [%s] [u:%s] [r:%s]" , c . nick , c . username , c . realname ) )
2017-05-08 01:15:16 +02:00
server . snomasks . Send ( sno . LocalConnects , fmt . Sprintf ( ircfmt . Unescape ( "Client registered $c[grey][$r%s$c[grey]] [u:$r%s$c[grey]] [h:$r%s$c[grey]] [r:$r%s$c[grey]]" ) , c . nick , c . username , c . rawHostname , c . realname ) )
2014-03-13 01:52:25 +01:00
c . Register ( )
2016-06-19 13:59:18 +02:00
// send welcome text
//NOTE(dan): we specifically use the NICK here instead of the nickmask
// see http://modern.ircdocs.horse/#rplwelcome-001 for details on why we avoid using the nickmask
2018-01-21 07:49:17 +01:00
c . Send ( nil , server . name , RPL_WELCOME , c . nick , fmt . Sprintf ( c . t ( "Welcome to the Internet Relay Network %s" ) , c . nick ) )
2018-01-23 07:50:19 +01:00
c . Send ( nil , server . name , RPL_YOURHOST , c . nick , fmt . Sprintf ( c . t ( "Your host is %[1]s, running version %[2]s" ) , server . name , Ver ) )
2018-01-22 12:26:01 +01:00
c . Send ( nil , server . name , RPL_CREATED , c . nick , fmt . Sprintf ( c . t ( "This server was created %s" ) , server . ctime . Format ( time . RFC1123 ) ) )
2016-06-19 13:59:18 +02:00
//TODO(dan): Look at adding last optional [<channel modes with a parameter>] parameter
2016-10-23 03:48:57 +02:00
c . Send ( nil , server . name , RPL_MYINFO , c . nick , server . name , Ver , supportedUserModesString , supportedChannelModesString )
2018-02-05 15:21:08 +01:00
2018-02-27 03:44:03 +01:00
rb = NewResponseBuffer ( c )
2018-02-05 15:21:08 +01:00
c . RplISupport ( rb )
server . MOTD ( c , rb )
rb . Send ( )
2018-04-09 08:37:06 +02:00
modestring := c . ModeString ( )
if modestring != "+" {
c . Send ( nil , c . nickMaskString , RPL_UMODEIS , c . nick , c . ModeString ( ) )
}
2017-10-04 19:41:19 +02:00
if server . logger . IsLoggingRawIO ( ) {
2018-01-22 12:26:01 +01:00
c . Notice ( 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." ) )
2017-03-08 12:57:31 +01:00
}
2018-01-21 04:13:20 +01:00
// if resumed, send fake channel joins
if c . resumeDetails != nil {
for _ , name := range c . resumeDetails . SendFakeJoinsFor {
channel := server . channels . Get ( name )
if channel == nil {
continue
}
if c . capabilities . Has ( caps . ExtendedJoin ) {
2018-02-11 11:30:40 +01:00
c . Send ( nil , c . nickMaskString , "JOIN" , channel . name , c . AccountName ( ) , c . realname )
2018-01-21 04:13:20 +01:00
} else {
c . Send ( nil , c . nickMaskString , "JOIN" , channel . name )
}
2018-02-05 15:21:08 +01:00
// reuse the last rb
channel . SendTopic ( c , rb )
channel . Names ( c , rb )
rb . Send ( )
2018-01-22 11:55:20 +01:00
// construct and send fake modestring if necessary
c . stateMutex . RLock ( )
myModes := channel . members [ c ]
c . stateMutex . RUnlock ( )
if myModes == nil {
continue
}
oldModes := myModes . String ( )
if 0 < len ( oldModes ) {
params := [ ] string { channel . name , "+" + oldModes }
2018-02-03 13:03:36 +01:00
for range oldModes {
2018-01-22 11:55:20 +01:00
params = append ( params , c . nick )
}
c . Send ( nil , server . name , "MODE" , params ... )
}
2018-01-21 04:13:20 +01:00
}
}
2012-12-15 23:34:20 +01:00
}
2018-01-21 07:49:17 +01:00
// t returns the translated version of the given string, based on the languages configured by the client.
func ( client * Client ) t ( originalString string ) string {
// grab this mutex to protect client.languages
client . stateMutex . RLock ( )
defer client . stateMutex . RUnlock ( )
return client . server . languages . Translate ( client . languages , originalString )
}
2017-03-06 00:27:08 +01:00
// MOTD serves the Message of the Day.
2018-02-05 15:21:08 +01:00
func ( server * Server ) MOTD ( client * Client , rb * ResponseBuffer ) {
2017-09-28 07:30:53 +02:00
server . configurableStateMutex . RLock ( )
2017-10-02 10:42:50 +02:00
motdLines := server . motdLines
server . configurableStateMutex . RUnlock ( )
2017-09-28 07:30:53 +02:00
2017-10-02 10:42:50 +02:00
if len ( motdLines ) < 1 {
2018-02-05 15:21:08 +01:00
rb . Add ( nil , server . name , ERR_NOMOTD , client . nick , client . t ( "MOTD File is missing" ) )
2014-02-12 01:35:32 +01:00
return
}
2018-02-05 15:21:08 +01:00
rb . Add ( nil , server . name , RPL_MOTDSTART , client . nick , fmt . Sprintf ( client . t ( "- %s Message of the day - " ) , server . name ) )
2017-10-02 10:42:50 +02:00
for _ , line := range motdLines {
2018-02-05 15:21:08 +01:00
rb . Add ( nil , server . name , RPL_MOTD , client . nick , line )
2014-02-12 01:35:32 +01:00
}
2018-02-05 15:21:08 +01:00
rb . Add ( nil , server . name , RPL_ENDOFMOTD , client . nick , client . t ( "End of MOTD command" ) )
2014-02-12 00:33:02 +01:00
}
2017-05-09 13:09:44 +02:00
// wordWrap wraps the given text into a series of lines that don't exceed lineWidth characters.
2017-01-14 06:28:50 +01:00
func wordWrap ( text string , lineWidth int ) [ ] string {
2017-01-17 23:05:31 +01:00
var lines [ ] string
var cacheLine , cacheWord string
for _ , char := range text {
2017-05-09 13:09:44 +02:00
if char == '\r' {
continue
} else if char == '\n' {
cacheLine += cacheWord
lines = append ( lines , cacheLine )
cacheWord = ""
cacheLine = ""
} else if ( char == ' ' || char == '-' ) && len ( cacheLine ) + len ( cacheWord ) + 1 < lineWidth {
// natural word boundary
2017-01-17 23:05:31 +01:00
cacheLine += cacheWord + string ( char )
cacheWord = ""
2017-05-09 13:09:44 +02:00
} else if lineWidth <= len ( cacheLine ) + len ( cacheWord ) + 1 {
// time to wrap to next line
2017-01-17 23:05:31 +01:00
if len ( cacheLine ) < ( lineWidth / 2 ) {
2017-05-09 13:09:44 +02:00
// this word takes up more than half a line... just split in the middle of the word
2017-01-17 23:05:31 +01:00
cacheLine += cacheWord + string ( char )
cacheWord = ""
2017-05-09 13:09:44 +02:00
} else {
cacheWord += string ( char )
2017-01-17 23:05:31 +01:00
}
lines = append ( lines , cacheLine )
cacheLine = ""
2017-01-14 06:28:50 +01:00
} else {
2017-05-09 13:09:44 +02:00
// normal character
2017-01-17 23:05:31 +01:00
cacheWord += string ( char )
2017-01-14 06:28:50 +01:00
}
}
2017-05-09 13:09:44 +02:00
if 0 < len ( cacheWord ) {
2017-01-17 23:05:31 +01:00
cacheLine += cacheWord
}
2017-05-09 13:09:44 +02:00
if 0 < len ( cacheLine ) {
2017-01-17 23:05:31 +01:00
lines = append ( lines , cacheLine )
}
2017-01-14 06:28:50 +01:00
2017-01-17 23:05:31 +01:00
return lines
2017-01-14 06:28:50 +01:00
}
// SplitMessage represents a message that's been split for sending.
type SplitMessage struct {
For512 [ ] string
ForMaxLine string
}
2017-01-14 12:48:57 +01:00
func ( server * Server ) splitMessage ( original string , origIs512 bool ) SplitMessage {
2017-01-14 06:28:50 +01:00
var newSplit SplitMessage
newSplit . ForMaxLine = original
2017-01-17 23:05:31 +01:00
if ! origIs512 {
2017-01-14 06:28:50 +01:00
newSplit . For512 = wordWrap ( original , 400 )
} else {
newSplit . For512 = [ ] string { original }
}
return newSplit
}
2017-03-06 00:27:08 +01:00
// WhoisChannelsNames returns the common channel names between two users.
2016-04-14 14:33:38 +02:00
func ( client * Client ) WhoisChannelsNames ( target * Client ) [ ] string {
2018-04-17 17:11:12 +02:00
isMultiPrefix := client . capabilities . Has ( caps . MultiPrefix )
2016-04-14 14:33:38 +02:00
var chstrs [ ] string
2018-04-17 17:11:12 +02:00
for _ , channel := range target . Channels ( ) {
2016-04-14 14:33:38 +02:00
// channel is secret and the target can't see it
2018-04-17 17:11:12 +02:00
if ! client . flags [ modes . Operator ] {
if ( target . HasMode ( modes . Invisible ) || channel . HasMode ( modes . Secret ) ) && ! channel . hasClient ( client ) {
continue
}
2016-04-14 14:33:38 +02:00
}
2018-04-17 17:11:12 +02:00
chstrs = append ( chstrs , channel . ClientPrefixes ( target , isMultiPrefix ) + channel . name )
2014-02-20 07:20:34 +01:00
}
return chstrs
}
2017-10-08 03:05:05 +02:00
func ( client * Client ) getWhoisOf ( target * Client , rb * ResponseBuffer ) {
2018-01-22 08:30:31 +01:00
target . stateMutex . RLock ( )
defer target . stateMutex . RUnlock ( )
2017-10-08 03:05:05 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISUSER , client . nick , target . nick , target . username , target . hostname , "*" , target . realname )
2017-01-13 02:05:58 +01:00
whoischannels := client . WhoisChannelsNames ( target )
if whoischannels != nil {
2017-10-08 03:05:05 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISCHANNELS , client . nick , target . nick , strings . Join ( whoischannels , " " ) )
2016-09-05 06:23:57 +02:00
}
2016-10-23 03:01:05 +02:00
if target . class != nil {
2017-10-08 03:05:05 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISOPERATOR , client . nick , target . nick , target . whoisLine )
2016-06-19 13:59:18 +02:00
}
2018-02-03 11:21:32 +01:00
if client . flags [ modes . Operator ] || client == target {
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISACTUALLY , client . nick , target . nick , fmt . Sprintf ( "%s@%s" , target . username , utils . LookupHostname ( target . IPString ( ) ) ) , target . IPString ( ) , client . t ( "Actual user@host, Actual IP" ) )
2017-06-22 21:15:10 +02:00
}
2018-02-03 11:21:32 +01:00
if target . flags [ modes . TLS ] {
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISSECURE , client . nick , target . nick , client . t ( "is using a secure connection" ) )
2017-06-22 21:15:10 +02:00
}
2018-02-11 11:30:40 +01:00
if target . LoggedIntoAccount ( ) {
2018-03-16 01:41:49 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISACCOUNT , client . nick , target . AccountName ( ) , client . t ( "is logged in as" ) )
2018-01-21 03:23:47 +01:00
}
2018-02-03 11:21:32 +01:00
if target . flags [ modes . Bot ] {
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISBOT , client . nick , target . nick , ircfmt . Unescape ( fmt . Sprintf ( client . t ( "is a $bBot$b on %s" ) , client . server . networkName ) ) )
2018-01-07 03:56:51 +01:00
}
2018-01-22 08:30:31 +01:00
if 0 < len ( target . languages ) {
params := [ ] string { client . nick , target . nick }
for _ , str := range client . server . languages . Codes ( target . languages ) {
params = append ( params , str )
}
2018-01-22 12:26:01 +01:00
params = append ( params , client . t ( "can speak these languages" ) )
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISLANGUAGE , params ... )
2018-01-22 08:30:31 +01:00
}
2018-02-03 11:21:32 +01:00
if target . certfp != "" && ( client . flags [ modes . Operator ] || client == target ) {
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISCERTFP , client . nick , target . nick , fmt . Sprintf ( client . t ( "has client certificate fingerprint %s" ) , target . certfp ) )
2016-06-19 13:59:18 +02:00
}
2018-01-27 15:30:30 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISIDLE , client . nick , target . nick , strconv . FormatUint ( target . IdleSeconds ( ) , 10 ) , strconv . FormatInt ( target . SignonTime ( ) , 10 ) , client . t ( "seconds idle, signon time" ) )
2016-06-19 13:59:18 +02:00
}
2017-10-23 01:50:16 +02:00
// rplWhoReply returns the WHO reply between one user and another channel/user.
2016-06-19 13:59:18 +02:00
// <channel> <user> <host> <server> <nick> ( "H" / "G" ) ["*"] [ ( "@" / "+" ) ]
// :<hopcount> <real name>
2018-02-05 15:21:08 +01:00
func ( target * Client ) rplWhoReply ( channel * Channel , client * Client , rb * ResponseBuffer ) {
2016-06-19 13:59:18 +02:00
channelName := "*"
flags := ""
2018-02-03 11:21:32 +01:00
if client . HasMode ( modes . Away ) {
2016-06-19 13:59:18 +02:00
flags = "G"
} else {
flags = "H"
}
2018-02-03 11:21:32 +01:00
if client . HasMode ( modes . Operator ) {
2016-06-19 13:59:18 +02:00
flags += "*"
}
if channel != nil {
2017-10-23 01:50:16 +02:00
flags += channel . ClientPrefixes ( client , target . capabilities . Has ( caps . MultiPrefix ) )
2016-10-11 15:51:46 +02:00
channelName = channel . name
2016-06-19 13:59:18 +02:00
}
2018-02-05 15:21:08 +01:00
rb . Add ( nil , target . server . name , RPL_WHOREPLY , target . nick , channelName , client . Username ( ) , client . Hostname ( ) , client . server . name , client . Nick ( ) , flags , strconv . Itoa ( client . hops ) + " " + client . Realname ( ) )
2014-02-09 02:43:59 +01:00
}
2014-02-09 03:14:39 +01:00
2018-02-05 15:21:08 +01:00
func whoChannel ( client * Client , channel * Channel , friends ClientSet , rb * ResponseBuffer ) {
2017-10-23 01:50:16 +02:00
for _ , member := range channel . Members ( ) {
2018-02-03 11:21:32 +01:00
if ! client . flags [ modes . Invisible ] || friends [ client ] {
2018-02-05 15:21:08 +01:00
client . rplWhoReply ( channel , member , rb )
2014-02-18 06:30:14 +01:00
}
2014-02-09 03:49:52 +01:00
}
}
2016-10-22 13:20:08 +02:00
// rehash reloads the config and applies the changes from the config file.
func ( server * Server ) rehash ( ) error {
2017-03-10 13:02:08 +01:00
server . logger . Debug ( "rehash" , "Starting rehash" )
2017-03-09 10:07:35 +01:00
2016-10-22 12:54:04 +02:00
// only let one REHASH go on at a time
server . rehashMutex . Lock ( )
2017-03-09 10:07:35 +01:00
defer server . rehashMutex . Unlock ( )
2017-03-10 13:02:08 +01:00
server . logger . Debug ( "rehash" , "Got rehash lock" )
2016-10-22 12:54:04 +02:00
2016-10-19 13:38:31 +02:00
config , err := LoadConfig ( server . configFilename )
2017-09-28 07:30:53 +02:00
if err != nil {
return fmt . Errorf ( "Error loading config file config: %s" , err . Error ( ) )
}
err = server . applyConfig ( config , false )
if err != nil {
return fmt . Errorf ( "Error applying config changes: %s" , err . Error ( ) )
}
return nil
}
func ( server * Server ) applyConfig ( config * Config , initial bool ) error {
if initial {
server . ctime = time . Now ( )
server . configFilename = config . Filename
2017-09-28 08:58:09 +02:00
} else {
// enforce configs that can't be changed after launch:
if server . limits . LineLen . Tags != config . Limits . LineLen . Tags || server . limits . LineLen . Rest != config . Limits . LineLen . Rest {
return fmt . Errorf ( "Maximum line length (linelen) cannot be changed after launching the server, rehash aborted" )
} else if server . name != config . Server . Name {
return fmt . Errorf ( "Server name cannot be changed after launching the server, rehash aborted" )
2018-02-19 03:52:39 +01:00
} else if server . storeFilename != config . Datastore . Path {
return fmt . Errorf ( "Datastore path cannot be changed after launching the server, rehash aborted" )
2017-09-28 08:58:09 +02:00
}
2017-09-28 07:30:53 +02:00
}
2016-10-19 13:38:31 +02:00
2018-02-19 03:52:39 +01:00
server . logger . Info ( "rehash" , "Using config file" , server . configFilename )
2017-09-28 07:30:53 +02:00
casefoldedName , err := Casefold ( config . Server . Name )
2016-10-19 13:38:31 +02:00
if err != nil {
2017-09-28 07:30:53 +02:00
return fmt . Errorf ( "Server name isn't valid [%s]: %s" , config . Server . Name , err . Error ( ) )
2016-10-19 13:38:31 +02:00
}
2016-10-23 02:47:11 +02:00
// confirm operator stuff all exists and is fine
operclasses , err := config . OperatorClasses ( )
if err != nil {
2017-01-12 08:40:01 +01:00
return fmt . Errorf ( "Error rehashing config file operclasses: %s" , err . Error ( ) )
2016-10-23 02:47:11 +02:00
}
opers , err := config . Operators ( operclasses )
if err != nil {
2017-01-12 08:40:01 +01:00
return fmt . Errorf ( "Error rehashing config file opers: %s" , err . Error ( ) )
2016-10-23 02:47:11 +02:00
}
2017-10-04 06:57:03 +02:00
// TODO: support rehash of existing operator perms?
2016-10-23 02:47:11 +02:00
2017-09-28 07:30:53 +02:00
// sanity checks complete, start modifying server state
2017-10-02 10:42:50 +02:00
if initial {
server . name = config . Server . Name
server . nameCasefolded = casefoldedName
}
2017-09-28 07:30:53 +02:00
2017-10-02 10:42:50 +02:00
server . configurableStateMutex . Lock ( )
2017-10-26 10:19:01 +02:00
server . networkName = config . Network . Name
2017-09-28 07:30:53 +02:00
if config . Server . Password != "" {
server . password = config . Server . PasswordBytes ( )
} else {
server . password = nil
}
2017-10-15 08:18:14 +02:00
// apply new WebIRC command restrictions
server . webirc = config . Server . WebIRC
2017-09-28 23:03:47 +02:00
// apply new PROXY command restrictions
server . proxyAllowedFrom = config . Server . ProxyAllowedFrom
2017-10-26 10:19:01 +02:00
server . recoverFromErrors = true
if config . Debug . RecoverFromErrors != nil {
server . recoverFromErrors = * config . Debug . RecoverFromErrors
}
server . configurableStateMutex . Unlock ( )
2017-09-28 23:03:47 +02:00
2017-10-09 23:37:13 +02:00
err = server . connectionLimiter . ApplyConfig ( config . Server . ConnectionLimiter )
2017-10-09 07:47:04 +02:00
if err != nil {
return err
}
2016-10-23 15:05:00 +02:00
2017-10-09 23:37:13 +02:00
err = server . connectionThrottler . ApplyConfig ( config . Server . ConnectionThrottler )
2017-10-09 07:47:04 +02:00
if err != nil {
return err
2016-10-23 15:05:00 +02:00
}
2016-10-22 14:18:41 +02:00
// setup new and removed caps
2017-09-29 09:25:58 +02:00
addedCaps := caps . NewSet ( )
removedCaps := caps . NewSet ( )
updatedCaps := caps . NewSet ( )
2016-10-22 14:18:41 +02:00
2018-01-22 08:30:31 +01:00
// Translations
currentLanguageValue , _ := CapValues . Get ( caps . Languages )
langCodes := [ ] string { strconv . Itoa ( len ( config . Languages . Data ) + 1 ) , "en" }
for _ , info := range config . Languages . Data {
if info . Incomplete {
langCodes = append ( langCodes , "~" + info . Code )
} else {
langCodes = append ( langCodes , info . Code )
}
}
newLanguageValue := strings . Join ( langCodes , "," )
server . logger . Debug ( "rehash" , "Languages:" , newLanguageValue )
if currentLanguageValue != newLanguageValue {
updatedCaps . Add ( caps . Languages )
CapValues . Set ( caps . Languages , newLanguageValue )
}
2018-02-03 10:46:14 +01:00
lm := languages . NewManager ( config . Languages . Default , config . Languages . Data )
2018-01-23 12:23:29 +01:00
server . logger . Debug ( "rehash" , "Regenerating HELP indexes for new languages" )
GenerateHelpIndices ( lm )
server . languages = lm
2016-10-22 14:18:41 +02:00
// SASL
2018-02-11 11:30:40 +01:00
oldAccountConfig := server . AccountConfig ( )
2018-02-19 04:15:43 +01:00
authPreviouslyEnabled := oldAccountConfig != nil && oldAccountConfig . AuthenticationEnabled
2018-02-11 11:30:40 +01:00
if config . Accounts . AuthenticationEnabled && ! authPreviouslyEnabled {
2016-10-22 14:18:41 +02:00
// enabling SASL
2017-09-29 09:25:58 +02:00
SupportedCapabilities . Enable ( caps . SASL )
CapValues . Set ( caps . SASL , "PLAIN,EXTERNAL" )
addedCaps . Add ( caps . SASL )
2018-02-11 11:30:40 +01:00
} else if ! config . Accounts . AuthenticationEnabled && authPreviouslyEnabled {
2016-10-22 14:18:41 +02:00
// disabling SASL
2017-09-29 09:25:58 +02:00
SupportedCapabilities . Disable ( caps . SASL )
removedCaps . Add ( caps . SASL )
2016-10-22 14:18:41 +02:00
}
2018-02-11 11:30:40 +01:00
2018-02-18 10:46:14 +01:00
nickReservationPreviouslyDisabled := oldAccountConfig != nil && ! oldAccountConfig . NickReservation . Enabled
nickReservationNowEnabled := config . Accounts . NickReservation . Enabled
2018-02-11 11:30:40 +01:00
if nickReservationPreviouslyDisabled && nickReservationNowEnabled {
server . accounts . buildNickToAccountIndex ( )
}
2016-10-22 14:18:41 +02:00
2017-03-09 10:07:35 +01:00
// STS
stsValue := config . Server . STS . Value ( )
var stsDisabled bool
2017-09-29 09:25:58 +02:00
stsCurrentCapValue , _ := CapValues . Get ( caps . STS )
server . logger . Debug ( "rehash" , "STS Vals" , stsCurrentCapValue , stsValue , fmt . Sprintf ( "server[%v] config[%v]" , server . stsEnabled , config . Server . STS . Enabled ) )
2017-03-09 10:07:35 +01:00
if config . Server . STS . Enabled && ! server . stsEnabled {
// enabling STS
2017-09-29 09:25:58 +02:00
SupportedCapabilities . Enable ( caps . STS )
addedCaps . Add ( caps . STS )
CapValues . Set ( caps . STS , stsValue )
2017-03-09 10:07:35 +01:00
} else if ! config . Server . STS . Enabled && server . stsEnabled {
// disabling STS
2017-09-29 09:25:58 +02:00
SupportedCapabilities . Disable ( caps . STS )
removedCaps . Add ( caps . STS )
2017-03-09 10:07:35 +01:00
stsDisabled = true
2017-09-29 09:25:58 +02:00
} else if config . Server . STS . Enabled && server . stsEnabled && stsValue != stsCurrentCapValue {
2017-03-09 10:07:35 +01:00
// STS policy updated
2017-09-29 09:25:58 +02:00
CapValues . Set ( caps . STS , stsValue )
updatedCaps . Add ( caps . STS )
2017-03-09 10:07:35 +01:00
}
server . stsEnabled = config . Server . STS . Enabled
2016-10-22 14:18:41 +02:00
// burst new and removed caps
var capBurstClients ClientSet
2017-09-29 09:25:58 +02:00
added := make ( map [ caps . Version ] string )
2016-10-22 14:18:41 +02:00
var removed string
2017-03-09 10:07:35 +01:00
// updated caps get DEL'd and then NEW'd
// so, we can just add updated ones to both removed and added lists here and they'll be correctly handled
2017-09-29 09:25:58 +02:00
server . logger . Debug ( "rehash" , "Updated Caps" , updatedCaps . String ( caps . Cap301 , CapValues ) , strconv . Itoa ( updatedCaps . Count ( ) ) )
for _ , capab := range updatedCaps . List ( ) {
addedCaps . Enable ( capab )
removedCaps . Enable ( capab )
2017-03-09 10:07:35 +01:00
}
2017-09-29 09:25:58 +02:00
if 0 < addedCaps . Count ( ) || 0 < removedCaps . Count ( ) {
2017-09-29 04:07:52 +02:00
capBurstClients = server . clients . AllWithCaps ( caps . CapNotify )
2016-10-22 14:18:41 +02:00
2017-09-29 09:25:58 +02:00
added [ caps . Cap301 ] = addedCaps . String ( caps . Cap301 , CapValues )
added [ caps . Cap302 ] = addedCaps . String ( caps . Cap302 , CapValues )
// removed never has values, so we leave it as Cap301
removed = removedCaps . String ( caps . Cap301 , CapValues )
2016-10-22 14:18:41 +02:00
}
for sClient := range capBurstClients {
2017-03-09 10:07:35 +01:00
if stsDisabled {
// remove STS policy
//TODO(dan): this is an ugly hack. we can write this better.
stsPolicy := "sts=duration=0"
2017-09-29 09:25:58 +02:00
if 0 < addedCaps . Count ( ) {
added [ caps . Cap302 ] = added [ caps . Cap302 ] + " " + stsPolicy
2017-03-09 10:07:35 +01:00
} else {
2017-09-29 09:25:58 +02:00
addedCaps . Enable ( caps . STS )
added [ caps . Cap302 ] = stsPolicy
2017-03-09 10:07:35 +01:00
}
2016-10-22 14:18:41 +02:00
}
2017-03-09 10:09:58 +01:00
// DEL caps and then send NEW ones so that updated caps get removed/added correctly
2017-09-29 09:25:58 +02:00
if 0 < removedCaps . Count ( ) {
2016-10-22 14:18:41 +02:00
sClient . Send ( nil , server . name , "CAP" , sClient . nick , "DEL" , removed )
}
2017-09-29 09:25:58 +02:00
if 0 < addedCaps . Count ( ) {
2017-03-09 10:07:35 +01:00
sClient . Send ( nil , server . name , "CAP" , sClient . nick , "NEW" , added [ sClient . capVersion ] )
}
2016-10-22 14:18:41 +02:00
}
2016-10-19 13:38:31 +02:00
// set server options
2017-10-02 10:42:50 +02:00
server . configurableStateMutex . Lock ( )
2017-03-09 10:07:35 +01:00
lineLenConfig := LineLenLimits {
Tags : config . Limits . LineLen . Tags ,
Rest : config . Limits . LineLen . Rest ,
}
2016-10-19 13:38:31 +02:00
server . limits = Limits {
AwayLen : int ( config . Limits . AwayLen ) ,
ChannelLen : int ( config . Limits . ChannelLen ) ,
KickLen : int ( config . Limits . KickLen ) ,
MonitorEntries : int ( config . Limits . MonitorEntries ) ,
NickLen : int ( config . Limits . NickLen ) ,
TopicLen : int ( config . Limits . TopicLen ) ,
2016-10-23 16:50:18 +02:00
ChanListModes : int ( config . Limits . ChanListModes ) ,
2017-03-09 10:07:35 +01:00
LineLen : lineLenConfig ,
2016-10-19 13:38:31 +02:00
}
2016-10-23 02:47:11 +02:00
server . operclasses = * operclasses
server . operators = opers
2016-10-19 13:38:31 +02:00
server . checkIdent = config . Server . CheckIdent
// registration
2017-03-24 03:52:38 +01:00
server . channelRegistrationEnabled = config . Channels . Registration . Enabled
2016-10-19 13:38:31 +02:00
2017-09-06 23:34:38 +02:00
server . defaultChannelModes = ParseDefaultChannelModes ( config )
2017-09-28 07:30:53 +02:00
server . configurableStateMutex . Unlock ( )
2017-09-06 23:34:38 +02:00
2017-03-13 23:12:39 +01:00
// set new sendqueue size
2018-03-18 02:32:12 +01:00
server . SetMaxSendQBytes ( config . Server . MaxSendQBytes )
2017-03-13 23:12:39 +01:00
2017-10-08 12:17:49 +02:00
server . loadMOTD ( config . Server . MOTD , config . Server . MOTDFormatting )
2017-09-28 07:30:53 +02:00
2017-10-02 05:31:40 +02:00
// reload logging config
err = server . logger . ApplyConfig ( config . Logging )
if err != nil {
return err
}
2017-10-04 19:41:19 +02:00
nowLoggingRawIO := server . logger . IsLoggingRawIO ( )
// notify clients if raw i/o logging was enabled by a rehash
sendRawOutputNotice := ! initial && ! server . loggingRawIO && nowLoggingRawIO
server . loggingRawIO = nowLoggingRawIO
2017-10-02 05:31:40 +02:00
2018-03-22 16:04:21 +01:00
// save a pointer to the new config
server . configurableStateMutex . Lock ( )
server . config = config
server . configurableStateMutex . Unlock ( )
2018-02-19 03:52:39 +01:00
server . storeFilename = config . Datastore . Path
server . logger . Info ( "rehash" , "Using datastore" , server . storeFilename )
2017-09-28 07:30:53 +02:00
if initial {
2018-02-19 03:52:39 +01:00
if err := server . loadDatastore ( server . storeFilename ) ; err != nil {
2017-09-28 07:30:53 +02:00
return err
}
}
2018-03-13 19:46:39 +01:00
server . setupPprofListener ( config )
2018-03-22 16:04:21 +01:00
// set RPL_ISUPPORT
var newISupportReplies [ ] [ ] string
oldISupportList := server . ISupport ( )
server . setISupport ( )
if oldISupportList != nil {
newISupportReplies = oldISupportList . GetDifference ( server . ISupport ( ) )
}
2017-09-28 07:30:53 +02:00
// we are now open for business
server . setupListeners ( config )
2017-10-02 05:31:40 +02:00
if ! initial {
// push new info to all of our clients
2017-11-22 10:41:11 +01:00
for _ , sClient := range server . clients . AllClients ( ) {
2017-10-02 05:31:40 +02:00
for _ , tokenline := range newISupportReplies {
sClient . Send ( nil , server . name , RPL_ISUPPORT , append ( [ ] string { sClient . nick } , tokenline ... ) ... )
}
2017-10-04 19:41:19 +02:00
if sendRawOutputNotice {
2018-01-22 12:26:01 +01:00
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." ) )
2017-10-02 05:31:40 +02:00
}
}
}
2017-09-28 07:30:53 +02:00
return nil
}
2018-03-13 19:46:39 +01:00
func ( server * Server ) setupPprofListener ( config * Config ) {
pprofListener := ""
if config . Debug . PprofListener != nil {
pprofListener = * config . Debug . PprofListener
}
if server . pprofServer != nil {
if pprofListener == "" || ( pprofListener != server . pprofServer . Addr ) {
server . logger . Info ( "rehash" , "Stopping pprof listener" , server . pprofServer . Addr )
server . pprofServer . Close ( )
server . pprofServer = nil
}
}
if pprofListener != "" && server . pprofServer == nil {
ps := http . Server {
Addr : pprofListener ,
}
go func ( ) {
if err := ps . ListenAndServe ( ) ; err != nil {
server . logger . Error ( "rehash" , fmt . Sprintf ( "pprof listener failed: %v" , err ) )
}
} ( )
server . pprofServer = & ps
server . logger . Info ( "rehash" , "Started pprof listener" , server . pprofServer . Addr )
}
}
2017-10-08 12:17:49 +02:00
func ( server * Server ) loadMOTD ( motdPath string , useFormatting bool ) error {
2018-02-19 03:52:39 +01:00
server . logger . Info ( "rehash" , "Using MOTD" , motdPath )
2017-09-28 07:30:53 +02:00
motdLines := make ( [ ] string , 0 )
if motdPath != "" {
file , err := os . Open ( motdPath )
if err == nil {
defer file . Close ( )
reader := bufio . NewReader ( file )
for {
line , err := reader . ReadString ( '\n' )
if err != nil {
break
}
line = strings . TrimRight ( line , "\r\n" )
2017-10-08 12:17:49 +02:00
if useFormatting {
line = ircfmt . Unescape ( line )
}
2017-09-28 07:30:53 +02:00
// "- " is the required prefix for MOTD, we just add it here to make
// bursting it out to clients easier
line = fmt . Sprintf ( "- %s" , line )
motdLines = append ( motdLines , line )
}
} else {
return err
2016-10-19 13:38:31 +02:00
}
}
2017-09-28 07:30:53 +02:00
server . configurableStateMutex . Lock ( )
server . motdLines = motdLines
2017-10-02 10:42:50 +02:00
server . configurableStateMutex . Unlock ( )
2017-09-28 07:30:53 +02:00
return nil
}
func ( server * Server ) loadDatastore ( datastorePath string ) error {
// open the datastore and load server state for which it (rather than config)
// is the source of truth
db , err := OpenDatabase ( datastorePath )
if err == nil {
server . store = db
} else {
return fmt . Errorf ( "Failed to open datastore: %s" , err . Error ( ) )
}
// load *lines (from the datastores)
server . logger . Debug ( "startup" , "Loading D/Klines" )
server . loadDLines ( )
server . loadKLines ( )
// load password manager
server . logger . Debug ( "startup" , "Loading passwords" )
err = server . store . View ( func ( tx * buntdb . Tx ) error {
saltString , err := tx . Get ( keySalt )
if err != nil {
return fmt . Errorf ( "Could not retrieve salt string: %s" , err . Error ( ) )
}
salt , err := base64 . StdEncoding . DecodeString ( saltString )
if err != nil {
return err
}
2017-10-05 16:03:53 +02:00
pwm := passwd . NewSaltedManager ( salt )
2017-09-28 07:30:53 +02:00
server . passwords = & pwm
return nil
} )
if err != nil {
return fmt . Errorf ( "Could not load salt: %s" , err . Error ( ) )
}
2017-11-09 04:19:50 +01:00
server . channelRegistry = NewChannelRegistry ( server )
2018-02-11 11:30:40 +01:00
server . accounts = NewAccountManager ( server )
2017-09-28 07:30:53 +02:00
return nil
}
func ( server * Server ) setupListeners ( config * Config ) {
2018-02-01 21:40:20 +01:00
logListener := func ( addr string , tlsconfig * tls . Config ) {
server . logger . Info ( "listeners" ,
fmt . Sprintf ( "now listening on %s, tls=%t." , addr , ( tlsconfig != nil ) ) ,
)
}
2017-09-08 02:20:08 +02:00
// update or destroy all existing listeners
2016-10-22 12:54:04 +02:00
tlsListeners := config . TLSListeners ( )
for addr := range server . listeners {
2017-09-08 02:20:08 +02:00
currentListener := server . listeners [ addr ]
2017-09-12 00:40:15 +02:00
var stillConfigured bool
2016-10-22 12:54:04 +02:00
for _ , newaddr := range config . Server . Listen {
if newaddr == addr {
2017-09-12 00:40:15 +02:00
stillConfigured = true
2016-10-22 12:54:04 +02:00
break
}
}
2017-09-12 00:40:15 +02:00
// pass new config information to the listener, to be picked up after
// its next Accept(). this is like sending over a buffered channel of
// size 1, but where sending a second item overwrites the buffered item
// instead of blocking.
currentListener . configMutex . Lock ( )
currentListener . shouldStop = ! stillConfigured
currentListener . tlsConfig = tlsListeners [ addr ]
currentListener . configMutex . Unlock ( )
if stillConfigured {
2018-02-01 21:40:20 +01:00
logListener ( addr , currentListener . tlsConfig )
2016-10-22 12:54:04 +02:00
} else {
2017-09-12 00:40:15 +02:00
// tell the listener it should stop by interrupting its Accept() call:
currentListener . listener . Close ( )
2017-09-08 02:20:08 +02:00
delete ( server . listeners , addr )
2017-09-28 07:30:53 +02:00
server . logger . Info ( "listeners" , fmt . Sprintf ( "stopped listening on %s." , addr ) )
2016-10-22 12:54:04 +02:00
}
}
2017-09-08 02:20:08 +02:00
// create new listeners that were not previously configured
2016-10-22 12:54:04 +02:00
for _ , newaddr := range config . Server . Listen {
_ , exists := server . listeners [ newaddr ]
if ! exists {
// make new listener
2018-02-01 21:40:20 +01:00
tlsConfig := tlsListeners [ newaddr ]
server . listeners [ newaddr ] = server . createListener ( newaddr , tlsConfig )
logListener ( newaddr , tlsConfig )
2016-10-22 12:54:04 +02:00
}
}
2017-09-28 07:30:53 +02:00
if len ( tlsListeners ) == 0 {
server . logger . Warning ( "startup" , "You are not exposing an SSL/TLS listening port. You should expose at least one port (typically 6697) to accept TLS connections" )
}
var usesStandardTLSPort bool
for addr := range config . TLSListeners ( ) {
if strings . Contains ( addr , "6697" ) {
usesStandardTLSPort = true
break
}
}
if 0 < len ( tlsListeners ) && ! usesStandardTLSPort {
server . logger . Warning ( "startup" , "Port 6697 is the standard TLS port for IRC. You should (also) expose port 6697 as a TLS port to ensure clients can connect securely" )
}
}
2017-06-11 07:59:03 +02:00
// elistMatcher takes and matches ELIST conditions
type elistMatcher struct {
MinClientsActive bool
MinClients int
MaxClientsActive bool
MaxClients int
}
// Matches checks whether the given channel matches our matches.
func ( matcher * elistMatcher ) Matches ( channel * Channel ) bool {
if matcher . MinClientsActive {
2017-10-23 01:50:16 +02:00
if len ( channel . Members ( ) ) < matcher . MinClients {
2017-06-11 07:59:03 +02:00
return false
}
}
if matcher . MaxClientsActive {
2017-10-23 01:50:16 +02:00
if len ( channel . Members ( ) ) < len ( channel . members ) {
2017-06-11 07:59:03 +02:00
return false
}
}
return true
}
// RplList returns the RPL_LIST numeric for the given channel.
2018-02-05 15:21:08 +01:00
func ( target * Client ) RplList ( channel * Channel , rb * ResponseBuffer ) {
2016-06-19 13:59:18 +02:00
// get the correct number of channel members
var memberCount int
2018-02-03 11:21:32 +01:00
if target . flags [ modes . Operator ] || channel . hasClient ( target ) {
2017-10-23 01:50:16 +02:00
memberCount = len ( channel . Members ( ) )
2016-06-19 13:59:18 +02:00
} else {
2017-10-23 01:50:16 +02:00
for _ , member := range channel . Members ( ) {
2018-02-03 11:21:32 +01:00
if ! member . HasMode ( modes . Invisible ) {
2017-03-06 00:27:08 +01:00
memberCount ++
2016-06-19 13:59:18 +02:00
}
}
}
2018-02-05 15:21:08 +01:00
rb . Add ( nil , target . server . name , RPL_LIST , target . nick , channel . name , strconv . Itoa ( memberCount ) , channel . topic )
2014-02-18 06:02:03 +01:00
}
2014-02-23 19:04:31 +01:00
2018-01-21 02:23:33 +01:00
// ResumeDetails are the details that we use to resume connections.
type ResumeDetails struct {
2018-01-21 04:13:20 +01:00
OldNick string
Timestamp * time . Time
SendFakeJoinsFor [ ] string
2018-01-21 02:23:33 +01:00
}
2017-10-29 08:59:56 +01:00
var (
2018-01-23 06:06:55 +01:00
infoString1 = strings . Split ( ` ▄ ▄ ▄ ▄ ▄ ▄ · ▄ ▄ • ▐ ▄
2017-10-29 08:59:56 +01:00
▪ ▀ ▄ █ · ▐ █ ▀ █ ▐ █ ▀ ▪ ▪ • █ ▌ ▐ █ ▪
2018-02-03 10:30:23 +01:00
▄ █ ▀ ▄ ▐ ▀ ▀ ▄ ▄ █ ▀ ▀ █ ▄ █ ▀ █ ▄ ▄ █ ▀ ▄ ▪ ▐ █ ▐ ▐ ▌ ▄ █ ▀ ▄
2017-10-29 08:59:56 +01:00
▐ █ ▌ . ▐ ▌ ▐ █ • █ ▌ ▐ █ ▪ ▐ ▌ ▐ █ ▄ ▪ ▐ █ ▐ █ ▌ ▐ ▌ █ █ ▐ █ ▌ ▐ █ ▌ . ▐ ▌
▀ █ ▄ ▀ ▪ . ▀ ▀ ▀ ▀ · ▀ ▀ ▀ ▀ ▀ █ ▄ ▀ ▀ ▀ █ ▪ ▀ █ ▄ ▀ ▪
https : //oragono.io/
https : //github.com/oragono/oragono
2018-01-23 08:22:39 +01:00
https : //crowdin.com/project/oragono
2018-01-23 06:06:55 +01:00
` , "\n" )
infoString2 = strings . Split ( ` Daniel Oakley , DanielOaks , < daniel @ danieloaks . net >
2017-10-29 08:59:56 +01:00
Shivaram Lingamneni , slingamn , < slingamn @ cs . stanford . edu >
2018-01-23 06:06:55 +01:00
` , "\n" )
infoString3 = strings . Split ( ` 3 onyc
2017-10-29 08:59:56 +01:00
Edmund Huber
Euan Kemp ( euank )
Jeremy Latt
Martin Lindhe ( martinlindhe )
Roberto Besser ( besser )
Robin Burchell ( rburchell )
Sean Enck ( enckse )
soul9
Vegax
` , "\n" )
)