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-09 02:10:04 +01:00
"fmt"
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"
2022-12-11 06:53:12 +01:00
"runtime/pprof"
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"
2022-08-03 06:59:00 +02:00
"sync/atomic"
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
2021-06-18 08:41:57 +02:00
"github.com/ergochat/irc-go/ircfmt"
2021-07-04 13:41:59 +02:00
"github.com/okzk/sdnotify"
2023-01-04 11:06:21 +01:00
"github.com/tidwall/buntdb"
2020-07-01 10:14:30 +02:00
2023-01-04 11:06:21 +01:00
"github.com/ergochat/ergo/irc/bunt"
2021-05-25 06:34:38 +02:00
"github.com/ergochat/ergo/irc/caps"
"github.com/ergochat/ergo/irc/connection_limits"
2023-01-04 11:06:21 +01:00
"github.com/ergochat/ergo/irc/datastore"
2021-05-25 06:34:38 +02:00
"github.com/ergochat/ergo/irc/flatip"
2022-01-02 00:56:40 +01:00
"github.com/ergochat/ergo/irc/flock"
2021-05-25 06:34:38 +02:00
"github.com/ergochat/ergo/irc/history"
"github.com/ergochat/ergo/irc/logger"
"github.com/ergochat/ergo/irc/modes"
"github.com/ergochat/ergo/irc/mysql"
"github.com/ergochat/ergo/irc/sno"
"github.com/ergochat/ergo/irc/utils"
2025-01-14 03:47:21 +01:00
"github.com/ergochat/ergo/irc/webpush"
2016-06-17 14:17:42 +02:00
)
2014-03-08 23:20:36 +01:00
2020-12-21 11:11:50 +01:00
const (
2022-03-30 21:35:28 +02:00
alwaysOnMaintenanceInterval = 30 * time . Minute
2025-01-14 03:47:21 +01:00
pushMaintenanceInterval = 24 * time . Hour
2020-12-21 11:11:50 +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
2019-02-26 03:50:43 +01:00
errorMsg = "ERROR :%s\r\n"
2017-10-30 10:21:47 +01:00
2020-05-28 20:27:41 +02:00
// three final parameters of 004 RPL_MYINFO, enumerating our supported modes
rplMyInfo1 , rplMyInfo2 , rplMyInfo3 = modes . RplMyInfo ( )
2018-02-03 12:15:07 +01:00
2020-10-21 03:24:47 +02:00
// CHANMODES isupport token
chanmodesToken = modes . ChanmodesToken ( )
2019-08-27 06:51:09 +02:00
// whitelist of caps to serve on the STS-only listener. In particular,
// never advertise SASL, to discourage people from sending their passwords:
2020-07-02 09:38:20 +02:00
stsOnlyCaps = caps . NewSet ( caps . STS , caps . MessageTags , caps . ServerTime , caps . Batch , caps . LabeledResponse , caps . EchoMessage , caps . Nope )
2019-09-22 21:48:17 +02:00
// we only have standard channels for now. TODO: any updates to this
// will also need to be reflected in CasefoldChannel
chanTypes = "#"
2019-11-18 07:42:48 +01:00
throttleMessage = "You have attempted to connect too many times within a short duration. Wait a while, and you will be able to connect."
2017-09-28 07:30:53 +02:00
)
2016-11-06 04:47:13 +01:00
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 {
2022-05-06 04:34:43 +02:00
accepts AcceptManager
2019-11-18 07:42:48 +01:00
accounts AccountManager
channels ChannelManager
clients ClientManager
2022-08-03 06:59:00 +02:00
config atomic . Pointer [ Config ]
2019-11-18 07:42:48 +01:00
configFilename string
connectionLimiter connection_limits . Limiter
ctime time . Time
dlines * DLineManager
helpIndexManager HelpIndexManager
klines * KLineManager
2020-05-05 04:29:10 +02:00
listeners map [ string ] IRCListener
2019-11-18 07:42:48 +01:00
logger * logger . Manager
monitorManager MonitorManager
name string
nameCasefolded string
rehashMutex sync . Mutex // tier 4
rehashSignal chan os . Signal
pprofServer * http . Server
2021-07-04 23:58:48 +02:00
exitSignals chan os . Signal
2022-12-11 06:53:12 +01:00
tracebackSignal chan os . Signal
2019-11-18 07:42:48 +01:00
snomasks SnoManager
store * buntdb . DB
2023-01-04 11:06:21 +01:00
dstore datastore . Datastore
2020-02-19 01:38:42 +01:00
historyDB mysql . MySQL
2019-11-18 07:42:48 +01:00
torLimiter connection_limits . TorLimiter
whoWas WhoWasList
stats Stats
semaphores ServerSemaphores
2022-01-02 00:56:40 +01:00
flock flock . Flocker
2025-01-12 05:07:04 +01:00
connIDCounter atomic . Uint64
2022-08-10 08:47:39 +02:00
defcon atomic . Uint32
2012-04-18 03:16:57 +02:00
}
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 ) {
2023-01-04 11:06:21 +01:00
// sanity check that kernel randomness is available; on modern Linux,
// this will block until it is, on other platforms it may panic:
utils . GenerateUUIDv4 ( )
2017-09-28 07:30:53 +02:00
// initialize data structures
2012-12-09 21:51:50 +01:00
server := & Server {
2022-12-11 06:53:12 +01:00
ctime : time . Now ( ) . UTC ( ) ,
listeners : make ( map [ string ] IRCListener ) ,
logger : logger ,
rehashSignal : make ( chan os . Signal , 1 ) ,
exitSignals : make ( chan os . Signal , len ( utils . ServerExitSignals ) ) ,
tracebackSignal : make ( chan os . Signal , len ( utils . ServerTracebackSignals ) ) ,
2012-12-15 23:34:20 +01:00
}
2022-08-10 08:47:39 +02:00
server . defcon . Store ( 5 )
2014-02-09 19:07:40 +01:00
2022-05-06 04:34:43 +02:00
server . accepts . Initialize ( )
2019-03-12 00:24:45 +01:00
server . clients . Initialize ( )
server . semaphores . Initialize ( )
server . whoWas . Initialize ( config . Limits . WhowasEntries )
2019-05-12 10:30:48 +02:00
server . monitorManager . Initialize ( )
server . snomasks . Initialize ( )
2019-02-12 06:27:57 +01:00
2020-02-19 01:38:42 +01:00
if err := server . applyConfig ( config ) ; err != nil {
2017-09-28 07:30:53 +02:00
return nil , err
2016-08-19 15:21:52 +02:00
}
2016-09-04 11:25:33 +02:00
2016-08-14 06:13:01 +02:00
// Attempt to clean up when receiving these signals.
2021-09-19 10:02:44 +02:00
signal . Notify ( server . exitSignals , utils . ServerExitSignals ... )
2016-10-22 13:20:08 +02:00
signal . Notify ( server . rehashSignal , syscall . SIGHUP )
2022-12-11 06:53:12 +01:00
if len ( utils . ServerTracebackSignals ) != 0 {
signal . Notify ( server . tracebackSignal , utils . ServerTracebackSignals ... )
}
2014-03-02 20:51:29 +01:00
2022-03-30 21:35:28 +02:00
time . AfterFunc ( alwaysOnMaintenanceInterval , server . periodicAlwaysOnMaintenance )
2025-01-14 03:47:21 +01:00
time . AfterFunc ( pushMaintenanceInterval , server . periodicPushMaintenance )
2020-12-21 11:11:50 +01:00
2017-03-06 06:50:23 +01:00
return server , nil
2016-10-19 13:38:31 +02: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 ( ) {
2021-07-04 13:41:59 +02:00
sdnotify . Stopping ( )
2021-07-04 23:58:48 +02:00
server . logger . Info ( "server" , "Stopping server" )
2021-07-04 13:41:59 +02:00
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
2022-03-30 21:35:28 +02:00
// flush data associated with always-on clients:
server . performAlwaysOnMaintenance ( false , true )
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
}
2020-02-19 01:38:42 +01:00
server . historyDB . Close ( )
2021-07-04 23:58:48 +02:00
server . logger . Info ( "server" , fmt . Sprintf ( "%s exiting" , Ver ) )
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 ( ) {
2021-07-04 23:58:48 +02:00
defer server . Shutdown ( )
2016-09-05 14:14:20 +02:00
2017-12-11 03:18:16 +01:00
for {
2014-02-17 07:38:43 +01:00
select {
2021-07-04 23:58:48 +02:00
case <- server . exitSignals :
2017-12-11 03:18:16 +01:00
return
2016-10-22 13:20:08 +02:00
case <- server . rehashSignal :
2021-03-11 05:04:16 +01:00
server . logger . Info ( "server" , "Rehashing due to SIGHUP" )
go server . rehash ( )
2022-12-11 06:53:12 +01:00
case <- server . tracebackSignal :
go server . dumpStacks ( )
2017-12-11 03:18:16 +01:00
}
}
}
2016-10-22 13:20:08 +02:00
2020-09-14 10:28:12 +02:00
func ( server * Server ) checkBans ( config * Config , ipaddr net . IP , checkScripts bool ) ( banned bool , requireSASL bool , message string ) {
2021-01-19 14:49:45 +01:00
// #671: do not enforce bans against loopback, as a failsafe
// note that this function is not used for Tor connections (checkTorLimits is used instead)
if ipaddr . IsLoopback ( ) {
return
}
2020-07-08 11:32:14 +02:00
if server . Defcon ( ) == 1 {
2021-01-19 14:49:45 +01:00
if ! utils . IPInNets ( ipaddr , server . Config ( ) . Server . secureNets ) {
2020-09-14 10:28:12 +02:00
return true , false , "New connections to this server are temporarily restricted"
2020-07-08 11:32:14 +02:00
}
}
2020-12-09 04:01:23 +01:00
flat := flatip . FromNetIP ( ipaddr )
2017-09-28 20:19:39 +02:00
// check DLINEs
2020-12-09 04:01:23 +01:00
isBanned , info := server . dlines . CheckIP ( flat )
2017-09-28 20:19:39 +02:00
if isBanned {
2021-01-22 15:38:40 +01:00
if info . RequireSASL {
server . logger . Info ( "connect-ip" , "Requiring SASL from client due to d-line" , ipaddr . String ( ) )
return false , true , info . BanMessage ( "You must authenticate with SASL to connect from this IP (%s)" )
} else {
server . logger . Info ( "connect-ip" , "Client rejected by d-line" , ipaddr . String ( ) )
return true , false , info . BanMessage ( "You are banned from this server (%s)" )
}
2017-09-28 20:19:39 +02:00
}
// check connection limits
2020-12-09 04:01:23 +01:00
err := server . connectionLimiter . AddClient ( flat )
2019-11-18 07:42:48 +01:00
if err == connection_limits . ErrLimitExceeded {
2017-09-28 20:19:39 +02:00
// too many connections from one client, tell the client and close the connection
2020-12-09 04:01:23 +01:00
server . logger . Info ( "connect-ip" , "Client rejected for connection limit" , ipaddr . String ( ) )
2020-09-14 10:28:12 +02:00
return true , false , "Too many clients from your network"
2019-11-18 07:42:48 +01:00
} else if err == connection_limits . ErrThrottleExceeded {
2020-12-09 04:01:23 +01:00
server . logger . Info ( "connect-ip" , "Client exceeded connection throttle" , ipaddr . String ( ) )
2020-09-14 10:28:12 +02:00
return true , false , throttleMessage
2019-11-18 07:42:48 +01:00
} else if err != nil {
server . logger . Warning ( "internal" , "unexpected ban result" , err . Error ( ) )
2017-09-28 20:19:39 +02:00
}
2022-01-02 07:51:31 +01:00
if checkScripts && config . Server . IPCheckScript . Enabled && ! config . Server . IPCheckScript . ExemptSASL {
2020-09-14 10:28:12 +02:00
output , err := CheckIPBan ( server . semaphores . IPCheckScript , config . Server . IPCheckScript , ipaddr )
if err != nil {
server . logger . Error ( "internal" , "couldn't check IP ban script" , ipaddr . String ( ) , err . Error ( ) )
return false , false , ""
}
2021-01-22 15:38:40 +01:00
// TODO: currently no way to cache IPAccepted
if ( output . Result == IPBanned || output . Result == IPRequireSASL ) && output . CacheSeconds != 0 {
2021-01-19 14:49:45 +01:00
network , err := flatip . ParseToNormalizedNet ( output . CacheNet )
2020-09-14 10:28:12 +02:00
if err != nil {
server . logger . Error ( "internal" , "invalid dline net from IP ban script" , ipaddr . String ( ) , output . CacheNet )
} else {
dlineDuration := time . Duration ( output . CacheSeconds ) * time . Second
2021-01-22 15:38:40 +01:00
err := server . dlines . AddNetwork ( network , dlineDuration , output . Result == IPRequireSASL , output . BanMessage , "" , "" )
2020-09-14 10:28:12 +02:00
if err != nil {
server . logger . Error ( "internal" , "couldn't set dline from IP ban script" , ipaddr . String ( ) , err . Error ( ) )
}
}
}
if output . Result == IPBanned {
// XXX roll back IP connection/throttling addition for the IP
2020-12-09 04:01:23 +01:00
server . connectionLimiter . RemoveClient ( flat )
2020-09-14 10:28:12 +02:00
server . logger . Info ( "connect-ip" , "Rejected client due to ip-check-script" , ipaddr . String ( ) )
return true , false , output . BanMessage
} else if output . Result == IPRequireSASL {
server . logger . Info ( "connect-ip" , "Requiring SASL from client due to ip-check-script" , ipaddr . String ( ) )
2020-09-14 14:11:56 +02:00
return false , true , output . BanMessage
2020-09-14 10:28:12 +02:00
}
}
return false , false , ""
2017-09-28 20:19:39 +02:00
}
2019-02-26 03:50:43 +01:00
func ( server * Server ) checkTorLimits ( ) ( banned bool , message string ) {
switch server . torLimiter . AddClient ( ) {
case connection_limits . ErrLimitExceeded :
return true , "Too many clients from the Tor network"
case connection_limits . ErrThrottleExceeded :
return true , "Exceeded connection throttle for the Tor network"
default :
return false , ""
}
}
2022-03-30 21:35:28 +02:00
func ( server * Server ) periodicAlwaysOnMaintenance ( ) {
2020-12-21 11:11:50 +01:00
defer func ( ) {
2021-09-19 03:28:16 +02:00
// reschedule whether or not there was a panic
2022-03-30 21:35:28 +02:00
time . AfterFunc ( alwaysOnMaintenanceInterval , server . periodicAlwaysOnMaintenance )
2020-12-21 11:11:50 +01:00
} ( )
2025-01-14 03:47:21 +01:00
defer server . HandlePanic ( nil )
2021-09-19 03:28:16 +02:00
2022-03-30 21:35:28 +02:00
server . logger . Info ( "accounts" , "Performing periodic always-on client checks" )
server . performAlwaysOnMaintenance ( true , true )
}
func ( server * Server ) performAlwaysOnMaintenance ( checkExpiration , flushTimestamps bool ) {
2020-12-21 11:11:50 +01:00
config := server . Config ( )
for _ , client := range server . clients . AllClients ( ) {
2022-03-30 21:35:28 +02:00
if checkExpiration && client . IsExpiredAlwaysOn ( config ) {
2020-12-21 11:11:50 +01:00
// TODO save the channels list, use it for autojoin if/when they return?
server . logger . Info ( "accounts" , "Expiring always-on client" , client . AccountName ( ) )
client . destroy ( nil )
2022-03-30 21:35:28 +02:00
continue
}
if flushTimestamps && client . shouldFlushTimestamps ( ) {
account := client . Account ( )
server . accounts . saveLastSeen ( account , client . copyLastSeen ( ) )
server . accounts . saveReadMarkers ( account , client . copyReadMarkers ( ) )
2020-12-21 11:11:50 +01:00
}
}
}
2025-01-14 03:47:21 +01:00
func ( server * Server ) periodicPushMaintenance ( ) {
defer func ( ) {
// reschedule whether or not there was a panic
time . AfterFunc ( pushMaintenanceInterval , server . periodicPushMaintenance )
} ( )
defer server . HandlePanic ( nil )
if server . Config ( ) . WebPush . Enabled {
server . logger . Info ( "webpush" , "Performing periodic push subscription maintenance" )
server . performPushMaintenance ( )
} // else: reschedule and check again later, the operator may enable it via rehash
}
func ( server * Server ) performPushMaintenance ( ) {
expiration := time . Duration ( server . Config ( ) . WebPush . Expiration )
for _ , client := range server . clients . AllWithPushSubscriptions ( ) {
2025-01-16 06:06:11 +01:00
for _ , sub := range client . getPushSubscriptions ( true ) {
2025-01-14 03:47:21 +01:00
now := time . Now ( )
// require both periodic successful push messages and renewal of the subscription via WEBPUSH REGISTER
if now . Sub ( sub . LastSuccess ) > expiration || now . Sub ( sub . LastRefresh ) > expiration {
server . logger . Debug ( "webpush" , "expiring push subscription for client" , client . Nick ( ) , sub . Endpoint )
client . deletePushSubscription ( sub . Endpoint , false )
} else if now . Sub ( sub . LastSuccess ) > expiration / 2 {
// we haven't pushed to them recently, make an attempt
server . logger . Debug ( "webpush" , "pinging push subscription for client" , client . Nick ( ) , sub . Endpoint )
client . sendAndTrackPush (
sub . Endpoint , sub . Keys ,
pushMessage {
msg : webpush . PingMessage ,
urgency : webpush . UrgencyNormal ,
} ,
false ,
)
}
}
// persist all push subscriptions on the assumption that the timestamps have changed
client . Store ( IncludePushSubscriptions )
}
}
2022-01-02 07:51:31 +01:00
// handles server.ip-check-script.exempt-sasl:
// run the ip check script at the end of the handshake, only for anonymous connections
func ( server * Server ) checkBanScriptExemptSASL ( config * Config , session * Session ) ( outcome AuthOutcome ) {
// TODO add caching for this; see related code in (*server).checkBans;
// we should probably just put an LRU around this instead of using the DLINE system
ipaddr := session . IP ( )
output , err := CheckIPBan ( server . semaphores . IPCheckScript , config . Server . IPCheckScript , ipaddr )
if err != nil {
server . logger . Error ( "internal" , "couldn't check IP ban script" , ipaddr . String ( ) , err . Error ( ) )
return authSuccess
}
if output . Result == IPBanned || output . Result == IPRequireSASL {
2025-01-12 05:07:04 +01:00
server . logger . Info ( "connect-ip" , session . connID , "Rejecting unauthenticated client due to ip-check-script" , ipaddr . String ( ) )
2022-01-02 07:51:31 +01:00
if output . BanMessage != "" {
session . client . requireSASLMessage = output . BanMessage
}
return authFailSaslRequired
}
return authSuccess
}
2012-12-09 07:54:58 +01:00
2019-05-23 02:25:57 +02:00
func ( server * Server ) tryRegister ( c * Client , session * Session ) ( exiting bool ) {
2020-09-14 10:28:12 +02:00
// XXX PROXY or WEBIRC MUST be sent as the first line of the session;
// if we are here at all that means we have the final value of the IP
2024-04-14 03:43:41 +02:00
c . finalizeHostname ( session )
2020-09-14 10:28:12 +02:00
2019-05-22 03:40:25 +02:00
// try to complete registration normally
2020-05-26 16:56:24 +02:00
// XXX(#1057) username can be filled in by an ident query without the client
// having sent USER: check for both username and realname to ensure they did
if c . preregNick == "" || c . username == "" || c . realname == "" || session . capState == caps . NegotiatingState {
2019-05-22 03:40:25 +02:00
return
}
2018-02-27 03:44:03 +01:00
2019-08-27 06:51:09 +02:00
if c . isSTSOnly {
2020-07-02 09:38:20 +02:00
server . playSTSBurst ( session )
2019-08-27 06:51:09 +02:00
return true
}
2019-05-22 03:40:25 +02:00
// client MUST send PASS if necessary, or authenticate with SASL if necessary,
// before completing the other registration commands
2020-07-08 11:32:14 +02:00
config := server . Config ( )
2020-09-14 10:28:12 +02:00
authOutcome := c . isAuthorized ( server , config , session , c . requireSASL )
2022-01-02 07:51:31 +01:00
if authOutcome == authSuccess && c . account == "" &&
config . Server . IPCheckScript . Enabled && config . Server . IPCheckScript . ExemptSASL {
authOutcome = server . checkBanScriptExemptSASL ( config , session )
}
2019-05-23 02:25:57 +02:00
var quitMessage string
switch authOutcome {
case authFailPass :
quitMessage = c . t ( "Password incorrect" )
c . Send ( nil , server . name , ERR_PASSWDMISMATCH , "*" , quitMessage )
case authFailSaslRequired , authFailTorSaslRequired :
2020-09-14 14:11:56 +02:00
quitMessage = c . requireSASLMessage
if quitMessage == "" {
quitMessage = c . t ( "You must log in with SASL to join this server" )
}
2019-05-23 02:25:57 +02:00
c . Send ( nil , c . server . name , "FAIL" , "*" , "ACCOUNT_REQUIRED" , quitMessage )
}
if authOutcome != authSuccess {
c . Quit ( quitMessage , nil )
return true
2019-05-22 03:40:25 +02:00
}
2020-09-14 14:11:56 +02:00
c . requireSASLMessage = ""
2018-02-27 03:44:03 +01:00
2019-05-22 03:40:25 +02:00
rb := NewResponseBuffer ( session )
2020-04-23 07:38:12 +02:00
nickError := performNickChange ( server , c , c , session , c . preregNick , rb )
2019-05-22 03:40:25 +02:00
rb . Send ( true )
2023-06-04 23:45:46 +02:00
if nickError != nil {
2019-05-22 03:40:25 +02:00
c . preregNick = ""
2020-04-23 07:38:12 +02:00
return false
2019-05-22 03:40:25 +02:00
}
2019-05-09 00:14:49 +02:00
if session . client != c {
// reattached, bail out.
// we'll play the reg burst later, on the new goroutine associated with
// (thisSession, otherClient). This is to avoid having to transfer state
// like nickname, hostname, etc. to show the correct values in the reg burst.
2020-04-23 07:38:12 +02:00
return false
2019-05-09 00:14:49 +02:00
}
2017-01-11 13:38:16 +01:00
2020-10-15 19:03:44 +02:00
// Apply default user modes (without updating the invisible counter)
// The number of invisible users will be updated by server.stats.Register
// if we're using default user mode +i.
for _ , defaultMode := range config . Accounts . defaultUserModes {
c . SetMode ( defaultMode , true )
}
2020-10-05 12:28:19 +02:00
// count new user in statistics (before checking KLINEs, see #1303)
server . stats . Register ( c . HasMode ( modes . Invisible ) )
2021-01-19 14:49:45 +01:00
// check KLINEs (#671: ignore KLINEs for loopback connections)
if ! session . IP ( ) . IsLoopback ( ) || session . isTor {
isBanned , info := server . klines . CheckMasks ( c . AllNickmasks ( ) ... )
2025-01-14 04:20:47 +01:00
if isBanned && ! ( info . RequireSASL && session . client . Account ( ) != "" ) {
2022-05-03 18:46:12 +02:00
c . setKlined ( )
2021-01-19 14:49:45 +01:00
c . Quit ( info . BanMessage ( c . t ( "You are banned from this server (%s)" ) ) , nil )
2025-01-12 05:07:04 +01:00
server . logger . Info ( "connect" , session . connID , "Client rejected by k-line" , c . NickMaskString ( ) )
2021-01-19 14:49:45 +01:00
return true
}
2019-12-17 21:10:23 +01:00
}
2019-05-09 00:14:49 +02:00
server . playRegistrationBurst ( session )
2023-07-05 03:44:18 +02:00
if len ( config . Channels . AutoJoin ) > 0 {
// only applicable to new clients, not reattaches:
server . handleAutojoins ( session , config . Channels . AutoJoin )
}
2019-05-23 02:25:57 +02:00
return false
2019-05-09 00:14:49 +02:00
}
2019-02-10 02:01:47 +01:00
2020-07-02 09:38:20 +02:00
func ( server * Server ) playSTSBurst ( session * Session ) {
nick := utils . SafeErrorParam ( session . client . preregNick )
session . Send ( nil , server . name , RPL_WELCOME , nick , fmt . Sprintf ( "Welcome to the Internet Relay Network %s" , nick ) )
2021-05-25 06:34:38 +02:00
session . Send ( nil , server . name , RPL_YOURHOST , nick , fmt . Sprintf ( "Your host is %[1]s, running version %[2]s" , server . name , "ergo" ) )
2020-07-02 09:38:20 +02:00
session . Send ( nil , server . name , RPL_CREATED , nick , fmt . Sprintf ( "This server was created %s" , time . Time { } . Format ( time . RFC1123 ) ) )
2021-05-25 06:34:38 +02:00
session . Send ( nil , server . name , RPL_MYINFO , nick , server . name , "ergo" , "o" , "o" , "o" )
2020-07-02 11:13:28 +02:00
session . Send ( nil , server . name , RPL_ISUPPORT , nick , "CASEMAPPING=ascii" , "are supported by this server" )
2020-07-02 09:38:20 +02:00
session . Send ( nil , server . name , ERR_NOMOTD , nick , "MOTD is unavailable" )
for _ , line := range server . Config ( ) . Server . STS . bannerLines {
session . Send ( nil , server . name , "NOTICE" , nick , line )
}
}
2019-05-09 00:14:49 +02:00
func ( server * Server ) playRegistrationBurst ( session * Session ) {
c := session . client
2017-01-11 13:38:16 +01:00
// continue registration
2019-05-08 10:11:54 +02:00
d := c . Details ( )
2025-01-12 05:07:04 +01:00
server . logger . Info ( "connect" , session . connID , fmt . Sprintf ( "Client connected [%s] [u:%s] [r:%s]" , d . nick , d . username , d . realname ) )
2020-02-19 01:38:42 +01:00
server . snomasks . Send ( sno . LocalConnects , fmt . Sprintf ( "Client connected [%s] [u:%s] [h:%s] [ip:%s] [r:%s]" , d . nick , d . username , session . rawHostname , session . IP ( ) . String ( ) , d . realname ) )
2020-11-30 02:20:26 +01:00
if d . account != "" {
server . sendLoginSnomask ( d . nickMask , d . accountName )
}
2018-11-26 11:23:27 +01:00
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
2020-11-29 22:34:36 +01:00
config := server . Config ( )
session . Send ( nil , server . name , RPL_WELCOME , d . nick , fmt . Sprintf ( c . t ( "Welcome to the %s IRC Network %s" ) , config . Network . Name , d . nick ) )
2019-05-09 00:14:49 +02:00
session . Send ( nil , server . name , RPL_YOURHOST , d . nick , fmt . Sprintf ( c . t ( "Your host is %[1]s, running version %[2]s" ) , server . name , Ver ) )
session . Send ( nil , server . name , RPL_CREATED , d . nick , fmt . Sprintf ( c . t ( "This server was created %s" ) , server . ctime . Format ( time . RFC1123 ) ) )
2020-05-28 20:27:41 +02:00
session . Send ( nil , server . name , RPL_MYINFO , d . nick , server . name , Ver , rplMyInfo1 , rplMyInfo2 , rplMyInfo3 )
2018-02-05 15:21:08 +01:00
2019-04-12 06:08:46 +02:00
rb := NewResponseBuffer ( session )
2024-09-27 06:40:56 +02:00
if ! ( rb . session . capabilities . Has ( caps . ExtendedISupport ) && rb . session . isupportSentPrereg ) {
server . RplISupport ( c , rb )
}
2022-07-17 04:34:21 +02:00
if d . account != "" && session . capabilities . Has ( caps . Persistence ) {
2022-08-04 03:37:10 +02:00
reportPersistenceStatus ( c , rb , false )
2022-07-15 06:04:35 +02:00
}
2019-07-01 15:21:38 +02:00
server . Lusers ( c , rb )
2018-02-05 15:21:08 +01:00
server . MOTD ( c , rb )
2018-12-28 19:45:55 +01:00
rb . Send ( true )
2018-02-05 15:21:08 +01:00
2018-04-09 08:37:06 +02:00
modestring := c . ModeString ( )
if modestring != "+" {
2020-04-16 08:48:01 +02:00
session . Send ( nil , server . name , RPL_UMODEIS , d . nick , modestring )
2018-04-09 08:37:06 +02:00
}
2019-12-19 12:33:43 +01:00
c . attemptAutoOper ( session )
2017-10-04 19:41:19 +02:00
if server . logger . IsLoggingRawIO ( ) {
2019-05-09 00:14:49 +02:00
session . Send ( nil , c . server . name , "NOTICE" , d . nick , c . t ( "This server is in debug mode and is logging all user I/O. If you do not wish for everything you send to be readable by the server owner(s), please disconnect." ) )
2019-02-10 02:01:47 +01:00
}
2012-12-15 23:34:20 +01:00
}
2019-07-01 15:21:38 +02:00
// RplISupport outputs our ISUPPORT lines to the client. This is used on connection and in VERSION responses.
func ( server * Server ) RplISupport ( client * Client , rb * ResponseBuffer ) {
2024-09-27 06:40:56 +02:00
server . sendRplISupportLines ( client , rb , server . Config ( ) . Server . isupport . CachedReply )
}
func ( server * Server ) sendRplISupportLines ( client * Client , rb * ResponseBuffer , lines [ ] [ ] string ) {
if rb . session . capabilities . Has ( caps . ExtendedISupport ) {
2024-10-27 03:11:20 +01:00
batchID := rb . StartNestedBatch ( caps . ExtendedISupportBatchType )
2024-09-27 06:40:56 +02:00
defer rb . EndNestedBatch ( batchID )
}
2019-07-01 15:21:38 +02:00
translatedISupport := client . t ( "are supported by this server" )
nick := client . Nick ( )
2024-09-27 06:40:56 +02:00
for _ , cachedTokenLine := range lines {
2019-07-01 15:21:38 +02:00
length := len ( cachedTokenLine ) + 2
tokenline := make ( [ ] string , length )
tokenline [ 0 ] = nick
copy ( tokenline [ 1 : ] , cachedTokenLine )
tokenline [ length - 1 ] = translatedISupport
rb . Add ( nil , server . name , RPL_ISUPPORT , tokenline ... )
}
}
func ( server * Server ) Lusers ( client * Client , rb * ResponseBuffer ) {
nick := client . Nick ( )
2021-11-01 09:48:31 +01:00
config := server . Config ( )
var stats StatsValues
var numChannels int
if ! config . Server . SuppressLusers || client . HasRoleCapabs ( "ban" ) {
stats = server . stats . GetValues ( )
numChannels = server . channels . Len ( )
}
2019-07-01 15:21:38 +02:00
rb . Add ( nil , server . name , RPL_LUSERCLIENT , nick , fmt . Sprintf ( client . t ( "There are %[1]d users and %[2]d invisible on %[3]d server(s)" ) , stats . Total - stats . Invisible , stats . Invisible , 1 ) )
rb . Add ( nil , server . name , RPL_LUSEROP , nick , strconv . Itoa ( stats . Operators ) , client . t ( "IRC Operators online" ) )
rb . Add ( nil , server . name , RPL_LUSERUNKNOWN , nick , strconv . Itoa ( stats . Unknown ) , client . t ( "unregistered connections" ) )
2021-11-01 09:48:31 +01:00
rb . Add ( nil , server . name , RPL_LUSERCHANNELS , nick , strconv . Itoa ( numChannels ) , client . t ( "channels formed" ) )
2020-03-18 10:36:48 +01:00
rb . Add ( nil , server . name , RPL_LUSERME , nick , fmt . Sprintf ( client . t ( "I have %[1]d clients and %[2]d servers" ) , stats . Total , 0 ) )
2019-07-01 15:21:38 +02:00
total := strconv . Itoa ( stats . Total )
max := strconv . Itoa ( stats . Max )
rb . Add ( nil , server . name , RPL_LOCALUSERS , nick , total , max , fmt . Sprintf ( client . t ( "Current local users %[1]s, max %[2]s" ) , total , max ) )
rb . Add ( nil , server . name , RPL_GLOBALUSERS , nick , total , max , fmt . Sprintf ( client . t ( "Current global users %[1]s, max %[2]s" ) , total , max ) )
2018-01-21 07:49:17 +01:00
}
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 ) {
2019-05-10 06:07:22 +02:00
motdLines := server . Config ( ) . Server . motdLines
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
}
2023-07-05 03:44:18 +02:00
func ( server * Server ) handleAutojoins ( session * Session , channelNames [ ] string ) {
rb := NewResponseBuffer ( session )
for _ , chname := range channelNames {
server . channels . Join ( session . client , chname , "" , false , rb )
}
rb . Send ( true )
}
2021-02-07 04:45:34 +01:00
func ( client * Client ) whoisChannelsNames ( target * Client , multiPrefix bool , hasPrivs bool ) [ ] string {
2016-04-14 14:33:38 +02:00
var chstrs [ ] string
2021-02-07 04:45:34 +01:00
targetInvis := target . HasMode ( modes . Invisible )
2018-04-17 17:11:12 +02:00
for _ , channel := range target . Channels ( ) {
2021-02-07 04:45:34 +01:00
if ! hasPrivs && ( targetInvis || channel . flags . HasMode ( modes . Secret ) ) && ! channel . hasClient ( client ) {
// client can't see *this* channel membership
continue
2016-04-14 14:33:38 +02:00
}
2019-04-12 06:08:46 +02:00
chstrs = append ( chstrs , channel . ClientPrefixes ( target , multiPrefix ) + channel . name )
2014-02-20 07:20:34 +01:00
}
return chstrs
}
2020-10-09 14:03:26 +02:00
func ( client * Client ) getWhoisOf ( target * Client , hasPrivs bool , rb * ResponseBuffer ) {
2021-02-07 04:45:34 +01:00
oper := client . Oper ( )
2018-08-17 17:13:38 +02:00
cnick := client . Nick ( )
2019-01-01 19:00:16 +01:00
targetInfo := target . Details ( )
rb . Add ( nil , client . server . name , RPL_WHOISUSER , cnick , targetInfo . nick , targetInfo . username , targetInfo . hostname , "*" , targetInfo . realname )
tnick := targetInfo . nick
2017-01-13 02:05:58 +01:00
2021-02-07 04:45:34 +01:00
whoischannels := client . whoisChannelsNames ( target , rb . session . capabilities . Has ( caps . MultiPrefix ) , oper . HasRoleCapab ( "sajoin" ) )
2017-01-13 02:05:58 +01:00
if whoischannels != nil {
2022-04-24 17:57:21 +02:00
for _ , line := range utils . BuildTokenLines ( maxLastArgLength , whoischannels , " " ) {
2022-04-24 08:47:31 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISCHANNELS , cnick , tnick , line )
}
2016-09-05 06:23:57 +02:00
}
2021-02-07 04:45:34 +01:00
if target . HasMode ( modes . Operator ) && operStatusVisible ( client , target , oper != nil ) {
2020-10-09 18:29:09 +02:00
tOper := target . Oper ( )
if tOper != nil {
rb . Add ( nil , client . server . name , RPL_WHOISOPERATOR , cnick , tnick , tOper . WhoisLine )
}
2016-06-19 13:59:18 +02:00
}
2021-02-07 04:45:34 +01:00
if client == target || oper . HasRoleCapab ( "ban" ) {
2021-08-26 04:31:38 +02:00
ip , hostname := target . getWhoisActually ( )
rb . Add ( nil , client . server . name , RPL_WHOISACTUALLY , cnick , tnick , fmt . Sprintf ( "%s@%s" , targetInfo . username , hostname ) , utils . IPStringToHostname ( ip . String ( ) ) , client . t ( "Actual user@host, Actual IP" ) )
2021-02-07 04:45:34 +01:00
}
if client == target || oper . HasRoleCapab ( "samode" ) {
2020-10-27 23:54:45 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISMODES , cnick , tnick , fmt . Sprintf ( client . t ( "is using modes +%s" ) , target . modes . String ( ) ) )
2017-06-22 21:15:10 +02:00
}
2018-04-23 00:47:10 +02:00
if target . HasMode ( modes . TLS ) {
2018-08-17 17:13:38 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISSECURE , cnick , tnick , client . t ( "is using a secure connection" ) )
2017-06-22 21:15:10 +02:00
}
2019-01-01 19:00:16 +01:00
if targetInfo . accountName != "*" {
rb . Add ( nil , client . server . name , RPL_WHOISACCOUNT , cnick , tnick , targetInfo . accountName , client . t ( "is logged in as" ) )
2018-01-21 03:23:47 +01:00
}
2018-04-23 00:47:10 +02:00
if target . HasMode ( modes . Bot ) {
2020-12-28 02:17:24 +01:00
rb . Add ( nil , client . server . name , RPL_WHOISBOT , cnick , tnick , fmt . Sprintf ( ircfmt . Unescape ( client . t ( "is a $bBot$b on %s" ) ) , client . server . Config ( ) . Network . Name ) )
2018-01-07 03:56:51 +01:00
}
2018-01-22 08:30:31 +01:00
2021-02-07 04:45:34 +01:00
if client == target || oper . HasRoleCapab ( "ban" ) {
2020-02-19 03:42:27 +01:00
for _ , session := range target . Sessions ( ) {
if session . certfp != "" {
rb . Add ( nil , client . server . name , RPL_WHOISCERTFP , cnick , tnick , fmt . Sprintf ( client . t ( "has client certificate fingerprint %s" ) , session . certfp ) )
}
}
2016-06-19 13:59:18 +02:00
}
2018-08-17 17:13:38 +02:00
rb . Add ( nil , client . server . name , RPL_WHOISIDLE , cnick , tnick , strconv . FormatUint ( target . IdleSeconds ( ) , 10 ) , strconv . FormatInt ( target . SignonTime ( ) , 10 ) , client . t ( "seconds idle, signon time" ) )
2020-07-17 07:55:13 +02:00
if away , awayMessage := target . Away ( ) ; away {
rb . Add ( nil , client . server . name , RPL_AWAY , cnick , tnick , awayMessage )
2020-03-20 20:14:27 +01:00
}
2016-06-19 13:59:18 +02: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 {
2021-03-11 05:04:16 +01:00
// #1570; this needs its own panic handling because it can be invoked via SIGHUP
2025-01-14 03:47:21 +01:00
defer server . HandlePanic ( nil )
2021-03-11 05:04:16 +01:00
2020-05-18 18:33:26 +02:00
server . logger . Info ( "server" , "Attempting 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 ( )
2021-07-04 13:41:59 +02:00
sdnotify . Reloading ( )
2021-07-05 01:51:35 +02:00
defer sdnotify . Ready ( )
2021-07-04 13:41:59 +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 {
2020-05-18 17:25:49 +02:00
server . logger . Error ( "server" , "failed to load config file" , err . Error ( ) )
return err
2017-09-28 07:30:53 +02:00
}
2020-02-19 01:38:42 +01:00
err = server . applyConfig ( config )
2017-09-28 07:30:53 +02:00
if err != nil {
2020-05-18 17:25:49 +02:00
server . logger . Error ( "server" , "Failed to rehash" , err . Error ( ) )
return err
2017-09-28 07:30:53 +02:00
}
2020-05-18 17:25:49 +02:00
server . logger . Info ( "server" , "Rehash completed successfully" )
2017-09-28 07:30:53 +02:00
return nil
}
2020-02-19 01:38:42 +01:00
func ( server * Server ) applyConfig ( config * Config ) ( err error ) {
oldConfig := server . Config ( )
initial := oldConfig == nil
2017-09-28 07:30:53 +02:00
if initial {
server . configFilename = config . Filename
2018-07-16 09:46:40 +02:00
server . name = config . Server . Name
server . nameCasefolded = config . Server . nameCasefolded
2019-12-18 13:06:04 +01:00
globalCasemappingSetting = config . Server . Casemapping
2020-06-22 20:54:43 +02:00
globalUtf8EnforcementSetting = config . Server . EnforceUtf8
2021-05-24 06:38:47 +02:00
MaxLineLen = config . Server . MaxLineLen
2017-09-28 08:58:09 +02:00
} else {
// enforce configs that can't be changed after launch:
2020-01-19 05:47:05 +01:00
if server . name != config . Server . Name {
2017-09-28 08:58:09 +02:00
return fmt . Errorf ( "Server name cannot be changed after launching the server, rehash aborted" )
2020-02-19 01:38:42 +01:00
} else if oldConfig . Datastore . Path != config . Datastore . Path {
2018-02-19 03:52:39 +01:00
return fmt . Errorf ( "Datastore path cannot be changed after launching the server, rehash aborted" )
2019-12-18 13:06:04 +01:00
} else if globalCasemappingSetting != config . Server . Casemapping {
return fmt . Errorf ( "Casemapping cannot be changed after launching the server, rehash aborted" )
2020-06-22 20:54:43 +02:00
} else if globalUtf8EnforcementSetting != config . Server . EnforceUtf8 {
return fmt . Errorf ( "UTF-8 enforcement cannot be changed after launching the server, rehash aborted" )
2020-02-26 07:44:05 +01:00
} else if oldConfig . Accounts . Multiclient . AlwaysOn != config . Accounts . Multiclient . AlwaysOn {
return fmt . Errorf ( "Default always-on setting cannot be changed after launching the server, rehash aborted" )
2020-09-09 10:01:46 +02:00
} else if oldConfig . Server . Relaymsg . Enabled != config . Server . Relaymsg . Enabled {
2020-06-08 07:17:45 +02:00
return fmt . Errorf ( "Cannot enable or disable relaying after launching the server, rehash aborted" )
2020-09-09 10:01:46 +02:00
} else if oldConfig . Server . Relaymsg . Separators != config . Server . Relaymsg . Separators {
2020-06-08 07:17:45 +02:00
return fmt . Errorf ( "Cannot change relaying separators after launching the server, rehash aborted" )
2020-09-14 10:28:12 +02:00
} else if oldConfig . Server . IPCheckScript . MaxConcurrency != config . Server . IPCheckScript . MaxConcurrency ||
oldConfig . Accounts . AuthScript . MaxConcurrency != config . Accounts . AuthScript . MaxConcurrency {
return fmt . Errorf ( "Cannot change max-concurrency for scripts after launching the server, rehash aborted" )
2020-11-29 05:40:21 +01:00
} else if oldConfig . Server . OverrideServicesHostname != config . Server . OverrideServicesHostname {
return fmt . Errorf ( "Cannot change override-services-hostname after launching the server, rehash aborted" )
2020-12-14 21:31:55 +01:00
} else if ! oldConfig . Datastore . MySQL . Enabled && config . Datastore . MySQL . Enabled {
return fmt . Errorf ( "Cannot enable MySQL after launching the server, rehash aborted" )
2021-05-24 06:38:47 +02:00
} else if oldConfig . Server . MaxLineLen != config . Server . MaxLineLen {
return fmt . Errorf ( "Cannot change max-line-len 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
2019-02-03 03:12:17 +01:00
server . logger . Info ( "server" , "Using config file" , server . configFilename )
2018-02-19 03:52:39 +01:00
2022-01-02 00:56:40 +01:00
if initial {
if config . LockFile != "" {
server . flock , err = flock . TryAcquireFlock ( config . LockFile )
if err != nil {
return fmt . Errorf ( "failed to acquire flock on %s: %w" ,
config . LockFile , err )
}
}
// the lock is never released until quit; we need to save a pointer
// to the (*flock.Flock) object so it doesn't get GC'ed, which would
// close the file and surrender the lock
}
2018-07-16 09:46:40 +02:00
// first, reload config sections for functionality implemented in subpackages:
wasLoggingRawIO := ! initial && server . logger . IsLoggingRawIO ( )
err = server . logger . ApplyConfig ( config . Logging )
2017-10-09 07:47:04 +02:00
if err != nil {
return err
2016-10-23 15:05:00 +02:00
}
2018-07-16 09:46:40 +02:00
nowLoggingRawIO := server . logger . IsLoggingRawIO ( )
// notify existing clients if raw i/o logging was enabled by a rehash
sendRawOutputNotice := ! wasLoggingRawIO && nowLoggingRawIO
2016-10-23 15:05:00 +02:00
2019-12-22 14:17:56 +01:00
server . connectionLimiter . ApplyConfig ( & config . Server . IPLimits )
tlConf := & config . Server . TorListeners
server . torLimiter . Configure ( tlConf . MaxConnections , tlConf . ThrottleDuration , tlConf . MaxConnectionsPerDuration )
2018-01-22 08:30:31 +01:00
// Translations
2019-02-19 08:54:57 +01:00
server . logger . Debug ( "server" , "Regenerating HELP indexes for new languages" )
server . helpIndexManager . GenerateIndices ( config . languageManager )
2018-01-22 08:30:31 +01:00
2020-09-14 10:28:12 +02:00
if initial {
maxIPConc := int ( config . Server . IPCheckScript . MaxConcurrency )
if maxIPConc != 0 {
2021-04-07 14:44:17 +02:00
server . semaphores . IPCheckScript = utils . NewSemaphore ( maxIPConc )
2020-09-14 10:28:12 +02:00
}
maxAuthConc := int ( config . Accounts . AuthScript . MaxConcurrency )
if maxAuthConc != 0 {
2021-04-07 14:44:17 +02:00
server . semaphores . AuthScript = utils . NewSemaphore ( maxAuthConc )
2020-09-14 10:28:12 +02:00
}
2020-11-29 05:40:21 +01:00
if err := overrideServicePrefixes ( config . Server . OverrideServicesHostname ) ; err != nil {
return err
}
2020-09-14 10:28:12 +02:00
}
2019-08-27 06:51:09 +02:00
if oldConfig != nil {
2019-12-22 14:17:56 +01:00
// if certain features were enabled by rehash, we need to load the corresponding data
// from the store
if ! oldConfig . Accounts . NickReservation . Enabled {
server . accounts . buildNickToAccountIndex ( config )
}
// resize history buffers as needed
2020-07-10 00:36:45 +02:00
if config . historyChangedFrom ( oldConfig ) {
2019-12-22 14:17:56 +01:00
for _ , channel := range server . channels . Channels ( ) {
2020-02-19 01:38:42 +01:00
channel . resizeHistory ( config )
2019-12-22 14:17:56 +01:00
}
for _ , client := range server . clients . AllClients ( ) {
2020-02-19 01:38:42 +01:00
client . resizeHistory ( config )
2019-12-22 14:17:56 +01:00
}
2018-11-26 11:23:27 +01:00
}
2020-03-27 22:52:37 +01:00
if oldConfig . Accounts . Registration . Throttling != config . Accounts . Registration . Throttling {
server . accounts . resetRegisterThrottle ( config )
}
2018-11-26 11:23:27 +01:00
}
2020-05-08 07:16:49 +02:00
server . logger . Info ( "server" , "Using datastore" , config . Datastore . Path )
if initial {
if err := server . loadDatastore ( config ) ; err != nil {
return err
}
} else {
if config . Datastore . MySQL . Enabled && config . Datastore . MySQL != oldConfig . Datastore . MySQL {
server . historyDB . SetConfig ( config . Datastore . MySQL )
}
}
// now that the datastore is initialized, we can load the cloak secret from it
// XXX this modifies config after the initial load, which is naughty,
// but there's no data race because we haven't done SetConfig yet
2023-01-04 11:06:21 +01:00
cloakSecret , err := LoadCloakSecret ( server . dstore )
if err != nil {
return fmt . Errorf ( "Could not load cloak secret: %w" , err )
}
config . Server . Cloaks . SetSecret ( cloakSecret )
2025-01-14 03:47:21 +01:00
// similarly bring the VAPID keys into the config, which requires regenerating the 005
if config . WebPush . Enabled {
config . WebPush . vapidKeys , err = LoadVAPIDKeys ( server . dstore )
if err != nil {
return fmt . Errorf ( "Could not load VAPID keys: %w" , err )
}
if err = config . generateISupport ( ) ; err != nil {
return fmt . Errorf ( "Could not regenerate cached 005 for VAPID: %w" , err )
}
}
2020-05-08 07:16:49 +02:00
2019-09-01 04:42:51 +02:00
// activate the new config
2022-08-03 06:59:00 +02:00
server . config . Store ( config )
2019-09-01 04:42:51 +02:00
2020-05-08 07:16:49 +02:00
// load [dk]-lines, registered users and channels, etc.
if initial {
if err := server . loadFromDatastore ( config ) ; err != nil {
return err
}
}
2016-10-22 14:18:41 +02:00
// burst new and removed caps
2019-12-23 21:26:37 +01:00
addedCaps , removedCaps := config . Diff ( oldConfig )
2019-04-12 06:08:46 +02:00
var capBurstSessions [ ] * Session
2019-08-27 06:51:09 +02:00
added := make ( map [ caps . Version ] [ ] string )
var removed [ ] string
2016-10-22 14:18:41 +02:00
2018-06-26 00:08:15 +02:00
if ! addedCaps . Empty ( ) || ! removedCaps . Empty ( ) {
2019-04-27 17:50:43 +02:00
capBurstSessions = server . clients . AllWithCapsNotify ( )
2016-10-22 14:18:41 +02:00
2019-11-10 02:31:56 +01:00
added [ caps . Cap301 ] = addedCaps . Strings ( caps . Cap301 , config . Server . capValues , 0 )
added [ caps . Cap302 ] = addedCaps . Strings ( caps . Cap302 , config . Server . capValues , 0 )
2017-09-29 09:25:58 +02:00
// removed never has values, so we leave it as Cap301
2019-11-10 02:31:56 +01:00
removed = removedCaps . Strings ( caps . Cap301 , config . Server . capValues , 0 )
2016-10-22 14:18:41 +02:00
}
2019-04-12 06:08:46 +02:00
for _ , sSession := range capBurstSessions {
2017-03-09 10:09:58 +01:00
// DEL caps and then send NEW ones so that updated caps get removed/added correctly
2018-06-26 00:08:15 +02:00
if ! removedCaps . Empty ( ) {
2019-08-27 06:51:09 +02:00
for _ , capStr := range removed {
sSession . Send ( nil , server . name , "CAP" , sSession . client . Nick ( ) , "DEL" , capStr )
}
2016-10-22 14:18:41 +02:00
}
2018-06-26 00:08:15 +02:00
if ! addedCaps . Empty ( ) {
2019-08-27 06:51:09 +02:00
for _ , capStr := range added [ sSession . capVersion ] {
sSession . Send ( nil , server . name , "CAP" , sSession . client . Nick ( ) , "NEW" , capStr )
}
2017-03-09 10:07:35 +01:00
}
2016-10-22 14:18:41 +02:00
}
2016-10-19 13:38:31 +02:00
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
2019-05-10 06:27:28 +02:00
if oldConfig != nil {
newISupportReplies = oldConfig . Server . isupport . GetDifference ( & config . Server . isupport )
2018-03-22 16:04:21 +01:00
}
2020-05-07 04:50:23 +02:00
if len ( config . Server . ProxyAllowedFrom ) != 0 {
server . logger . Info ( "server" , "Proxied IPs will be accepted from" , strings . Join ( config . Server . ProxyAllowedFrom , ", " ) )
}
2021-07-04 23:58:48 +02:00
// we are now ready to receive connections:
2018-08-06 11:02:47 +02:00
err = server . setupListeners ( config )
2017-09-28 07:30:53 +02:00
2021-07-05 01:51:35 +02:00
if initial && err == nil {
server . logger . Info ( "server" , "Server running" )
2021-07-04 23:58:48 +02:00
sdnotify . Ready ( )
}
2021-07-04 13:41:59 +02:00
2017-10-02 05:31:40 +02:00
if ! initial {
2024-09-27 06:40:56 +02:00
// send 005 updates (somewhat rare)
if len ( newISupportReplies ) != 0 {
for _ , sClient := range server . clients . AllClients ( ) {
for _ , session := range sClient . Sessions ( ) {
rb := NewResponseBuffer ( session )
server . sendRplISupportLines ( sClient , rb , newISupportReplies )
rb . Send ( false )
}
2017-10-02 05:31:40 +02:00
}
2024-09-27 06:40:56 +02:00
}
2017-10-02 05:31:40 +02:00
2024-09-27 06:40:56 +02:00
if sendRawOutputNotice {
for _ , sClient := range server . clients . AllClients ( ) {
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
}
}
}
2021-07-04 23:58:48 +02:00
// send other config warnings
if config . Accounts . RequireSasl . Enabled && config . Accounts . Registration . Enabled {
server . logger . Warning ( "server" , "Warning: although require-sasl is enabled, users can still register accounts. If your server is not intended to be public, you must set accounts.registration.enabled to false." )
}
2018-08-06 11:02:47 +02:00
return err
2017-09-28 07:30:53 +02:00
}
2018-03-13 19:46:39 +01:00
func ( server * Server ) setupPprofListener ( config * Config ) {
2021-09-19 08:09:43 +02:00
pprofListener := config . Debug . PprofListener
2018-03-13 19:46:39 +01:00
if server . pprofServer != nil {
if pprofListener == "" || ( pprofListener != server . pprofServer . Addr ) {
2019-02-03 03:12:17 +01:00
server . logger . Info ( "server" , "Stopping pprof listener" , server . pprofServer . Addr )
2018-03-13 19:46:39 +01:00
server . pprofServer . Close ( )
server . pprofServer = nil
}
}
if pprofListener != "" && server . pprofServer == nil {
ps := http . Server {
Addr : pprofListener ,
}
go func ( ) {
if err := ps . ListenAndServe ( ) ; err != nil {
2019-02-03 03:12:17 +01:00
server . logger . Error ( "server" , "pprof listener failed" , err . Error ( ) )
2018-03-13 19:46:39 +01:00
}
} ( )
server . pprofServer = & ps
2019-02-03 03:12:17 +01:00
server . logger . Info ( "server" , "Started pprof listener" , server . pprofServer . Addr )
2018-03-13 19:46:39 +01:00
}
}
2018-04-16 22:28:31 +02:00
func ( server * Server ) loadDatastore ( config * Config ) error {
2017-09-28 07:30:53 +02:00
// open the datastore and load server state for which it (rather than config)
// is the source of truth
2018-12-31 07:17:44 +01:00
_ , err := os . Stat ( config . Datastore . Path )
if os . IsNotExist ( err ) {
2019-02-03 03:12:17 +01:00
server . logger . Warning ( "server" , "database does not exist, creating it" , config . Datastore . Path )
2018-12-31 07:17:44 +01:00
err = initializeDB ( config . Datastore . Path )
if err != nil {
return err
}
}
2018-04-16 22:28:31 +02:00
db , err := OpenDatabase ( config )
2017-09-28 07:30:53 +02:00
if err == nil {
server . store = db
2023-01-04 11:06:21 +01:00
server . dstore = bunt . NewBuntdbDatastore ( db , server . logger )
2020-05-08 07:16:49 +02:00
return nil
2017-09-28 07:30:53 +02:00
} else {
return fmt . Errorf ( "Failed to open datastore: %s" , err . Error ( ) )
}
2020-05-08 07:16:49 +02:00
}
2017-09-28 07:30:53 +02:00
2020-05-08 07:16:49 +02:00
func ( server * Server ) loadFromDatastore ( config * Config ) ( err error ) {
2017-09-28 07:30:53 +02:00
// load *lines (from the datastores)
2019-02-03 03:12:17 +01:00
server . logger . Debug ( "server" , "Loading D/Klines" )
2017-09-28 07:30:53 +02:00
server . loadDLines ( )
server . loadKLines ( )
2023-01-04 11:06:21 +01:00
server . channels . Initialize ( server , config )
2019-03-12 00:24:45 +01:00
server . accounts . Initialize ( server )
2018-02-11 11:30:40 +01:00
2020-02-19 01:38:42 +01:00
if config . Datastore . MySQL . Enabled {
2020-02-21 00:33:48 +01:00
server . historyDB . Initialize ( server . logger , config . Datastore . MySQL )
err = server . historyDB . Open ( )
2020-02-19 01:38:42 +01:00
if err != nil {
server . logger . Error ( "internal" , "could not connect to mysql" , err . Error ( ) )
return err
}
}
2017-09-28 07:30:53 +02:00
return nil
}
2018-08-06 11:02:47 +02:00
func ( server * Server ) setupListeners ( config * Config ) ( err error ) {
2020-05-05 04:29:10 +02:00
logListener := func ( addr string , config utils . ListenerConfig ) {
2018-02-01 21:40:20 +01:00
server . logger . Info ( "listeners" ,
2020-11-19 18:31:58 +01:00
fmt . Sprintf ( "now listening on %s, tls=%t, proxy=%t, tor=%t, websocket=%t." , addr , ( config . TLSConfig != nil ) , config . RequireProxy , config . Tor , config . WebSocket ) ,
2018-02-01 21:40:20 +01:00
)
}
2017-09-08 02:20:08 +02:00
// update or destroy all existing listeners
2016-10-22 12:54:04 +02:00
for addr := range server . listeners {
2017-09-08 02:20:08 +02:00
currentListener := server . listeners [ addr ]
2019-06-28 16:45:34 +02:00
newConfig , stillConfigured := config . Server . trueListeners [ addr ]
2016-10-22 12:54:04 +02:00
2017-09-12 00:40:15 +02:00
if stillConfigured {
2020-05-07 12:42:50 +02:00
if reloadErr := currentListener . Reload ( newConfig ) ; reloadErr == nil {
logListener ( addr , newConfig )
} else {
// stop the listener; we will attempt to replace it below
2020-05-05 04:29:10 +02:00
currentListener . Stop ( )
2020-05-07 12:42:50 +02:00
delete ( server . listeners , addr )
2020-05-05 04:29:10 +02:00
}
2016-10-22 12:54:04 +02:00
} else {
2020-05-05 04:29:10 +02:00
currentListener . Stop ( )
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
}
}
2019-08-27 06:51:09 +02:00
publicPlaintextListener := ""
2020-05-07 12:42:50 +02:00
// create new listeners that were not previously configured,
// or that couldn't be reloaded above:
2019-06-28 16:45:34 +02:00
for newAddr , newConfig := range config . Server . trueListeners {
2019-11-20 23:43:40 +01:00
if strings . HasPrefix ( newAddr , ":" ) && ! newConfig . Tor && ! newConfig . STSOnly && newConfig . TLSConfig == nil {
2019-08-27 06:51:09 +02:00
publicPlaintextListener = newAddr
2019-06-28 16:45:34 +02:00
}
2019-06-18 04:21:37 +02:00
_ , exists := server . listeners [ newAddr ]
2016-10-22 12:54:04 +02:00
if ! exists {
2020-05-05 04:29:10 +02:00
// make a new listener
2020-05-05 23:20:50 +02:00
newListener , newErr := NewListener ( server , newAddr , newConfig , config . Server . UnixBindMode )
if newErr == nil {
2020-05-05 04:29:10 +02:00
server . listeners [ newAddr ] = newListener
logListener ( newAddr , newConfig )
2020-05-05 23:20:50 +02:00
} else {
server . logger . Error ( "server" , "couldn't listen on" , newAddr , newErr . Error ( ) )
err = newErr
2018-08-06 11:02:47 +02:00
}
2016-10-22 12:54:04 +02:00
}
}
2019-08-27 06:51:09 +02:00
if publicPlaintextListener != "" {
2021-04-27 14:44:54 +02:00
server . logger . Warning ( "listeners" , fmt . Sprintf ( "Warning: your server is configured with public plaintext listener %s. Consider disabling it for improved security and privacy." , publicPlaintextListener ) )
2017-09-28 07:30:53 +02:00
}
2018-08-06 11:02:47 +02:00
return
2017-09-28 07:30:53 +02:00
}
2020-02-19 01:38:42 +01:00
// Gets the abstract sequence from which we're going to query history;
// we may already know the channel we're querying, or we may have
2020-02-28 11:41:08 +01:00
// to look it up via a string query. This function is responsible for
2020-02-19 01:38:42 +01:00
// privilege checking.
2021-04-19 14:54:40 +02:00
// XXX: call this with providedChannel==nil and query=="" to get a sequence
// suitable for ListCorrespondents (i.e., this function is still used to
// decide whether the ringbuf or mysql is authoritative about the client's
// message history).
2021-11-01 06:23:07 +01:00
func ( server * Server ) GetHistorySequence ( providedChannel * Channel , client * Client , query string ) ( channel * Channel , sequence history . Sequence , err error ) {
2020-02-19 01:38:42 +01:00
config := server . Config ( )
2020-02-24 20:09:00 +01:00
// 4 cases: {persistent, ephemeral} x {normal, conversation}
2020-02-28 11:41:08 +01:00
// with ephemeral history, target is implicit in the choice of `hist`,
// and correspondent is "" if we're retrieving a channel or *, and the correspondent's name
2020-02-24 20:09:00 +01:00
// if we're retrieving a DM conversation ("query buffer"). with persistent history,
2020-02-28 11:41:08 +01:00
// target is always nonempty, and correspondent is either empty or nonempty as before.
2020-02-24 20:09:00 +01:00
var status HistoryStatus
2020-02-28 11:41:08 +01:00
var target , correspondent string
2020-02-19 01:38:42 +01:00
var hist * history . Buffer
2021-01-21 03:13:18 +01:00
restriction := HistoryCutoffNone
2020-02-24 20:09:00 +01:00
channel = providedChannel
if channel == nil {
2020-02-28 11:41:08 +01:00
if strings . HasPrefix ( query , "#" ) {
channel = server . channels . Get ( query )
2020-02-24 20:09:00 +01:00
if channel == nil {
2020-02-19 01:38:42 +01:00
return
}
2020-02-24 20:09:00 +01:00
}
}
2021-01-21 03:13:18 +01:00
var joinTimeCutoff time . Time
2020-02-24 20:09:00 +01:00
if channel != nil {
2021-01-21 03:13:18 +01:00
if present , cutoff := channel . joinTimeCutoff ( client ) ; present {
joinTimeCutoff = cutoff
} else {
2020-02-24 20:09:00 +01:00
err = errInsufficientPrivs
return
}
2021-01-21 03:13:18 +01:00
status , target , restriction = channel . historyStatus ( config )
2020-02-24 20:09:00 +01:00
switch status {
case HistoryEphemeral :
hist = & channel . history
case HistoryPersistent :
2020-02-28 11:41:08 +01:00
// already set `target`
2020-02-24 20:09:00 +01:00
default :
return
}
} else {
2020-02-28 11:41:08 +01:00
status , target = client . historyStatus ( config )
2021-04-19 14:54:40 +02:00
if query != "" {
2020-02-28 11:41:08 +01:00
correspondent , err = CasefoldName ( query )
2020-02-19 01:38:42 +01:00
if err != nil {
return
}
2020-02-24 20:09:00 +01:00
}
switch status {
case HistoryEphemeral :
hist = & client . history
case HistoryPersistent :
2020-02-28 11:41:08 +01:00
// already set `target`, and `correspondent` if necessary
2020-02-24 20:09:00 +01:00
default :
return
2020-02-19 01:38:42 +01:00
}
}
var cutoff time . Time
2022-05-12 22:43:11 +02:00
// #1593: cutoff is ignored for operators
if ! client . HasRoleCapabs ( "history" ) {
if config . History . Restrictions . ExpireTime != 0 {
cutoff = time . Now ( ) . UTC ( ) . Add ( - time . Duration ( config . History . Restrictions . ExpireTime ) )
}
// #836: registration date cutoff is always enforced for DMs
// either way, take the later of the two cutoffs
if restriction == HistoryCutoffRegistrationTime || channel == nil {
regCutoff := client . historyCutoff ( )
if regCutoff . After ( cutoff ) {
cutoff = regCutoff
}
} else if restriction == HistoryCutoffJoinTime {
if joinTimeCutoff . After ( cutoff ) {
cutoff = joinTimeCutoff
}
2021-01-21 03:13:18 +01:00
}
2022-05-12 22:43:11 +02:00
// #836 again: grace period is never applied to DMs
if ! cutoff . IsZero ( ) && channel != nil && restriction != HistoryCutoffJoinTime {
cutoff = cutoff . Add ( - time . Duration ( config . History . Restrictions . GracePeriod ) )
}
2020-02-24 11:44:55 +01:00
}
2020-02-24 20:09:00 +01:00
2020-02-19 01:38:42 +01:00
if hist != nil {
2021-11-01 06:23:07 +01:00
sequence = hist . MakeSequence ( correspondent , cutoff )
2020-02-28 11:41:08 +01:00
} else if target != "" {
2021-11-01 06:23:07 +01:00
sequence = server . historyDB . MakeSequence ( target , correspondent , cutoff )
2020-02-19 01:38:42 +01:00
}
return
}
2020-05-12 18:05:40 +02:00
func ( server * Server ) ForgetHistory ( accountName string ) {
// sanity check
if accountName == "*" {
return
}
config := server . Config ( )
if ! config . History . Enabled {
return
}
if cfAccount , err := CasefoldName ( accountName ) ; err == nil {
server . historyDB . Forget ( cfAccount )
}
persistent := config . History . Persistent
if persistent . Enabled && persistent . UnregisteredChannels && persistent . RegisteredChannels == PersistentMandatory && persistent . DirectMessages == PersistentMandatory {
return
}
predicate := func ( item * history . Item ) bool { return item . AccountName == accountName }
for _ , channel := range server . channels . Channels ( ) {
channel . history . Delete ( predicate )
}
for _ , client := range server . clients . AllClients ( ) {
client . history . Delete ( predicate )
}
}
// deletes a message. target is a hint about what buffer it's in (not required for
// persistent history, where all the msgids are indexed together). if accountName
// is anything other than "*", it must match the recorded AccountName of the message
func ( server * Server ) DeleteMessage ( target , msgid , accountName string ) ( err error ) {
config := server . Config ( )
var hist * history . Buffer
if target != "" {
if target [ 0 ] == '#' {
channel := server . channels . Get ( target )
if channel != nil {
2021-01-21 03:13:18 +01:00
if status , _ , _ := channel . historyStatus ( config ) ; status == HistoryEphemeral {
2020-05-12 18:05:40 +02:00
hist = & channel . history
}
}
} else {
client := server . clients . Get ( target )
if client != nil {
if status , _ := client . historyStatus ( config ) ; status == HistoryEphemeral {
hist = & client . history
}
}
}
}
if hist == nil {
err = server . historyDB . DeleteMsgid ( msgid , accountName )
} else {
count := hist . Delete ( func ( item * history . Item ) bool {
return item . Message . Msgid == msgid && ( accountName == "*" || item . AccountName == accountName )
} )
if count == 0 {
err = errNoop
}
}
return
}
2021-04-07 11:40:39 +02:00
func ( server * Server ) UnfoldName ( cfname string ) ( name string ) {
if strings . HasPrefix ( cfname , "#" ) {
return server . channels . UnfoldName ( cfname )
}
return server . clients . UnfoldNick ( cfname )
}
2025-01-12 05:07:04 +01:00
// generateConnectionID generates a unique string identifier for an incoming connection.
// this identifier is only used for debug logging.
func ( server * Server ) generateConnectionID ( ) string {
id := server . connIDCounter . Add ( 1 )
// pad with leading zeroes to a minimum length of 5 hex digits. this enhances greppability;
// the identifier length will be 6 for the first 1048576 connections, which is less important
// but makes the log slightly easier to read
return fmt . Sprintf ( "s%05x" , id )
}
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 {
2022-04-16 22:55:58 +02:00
if len ( channel . Members ( ) ) > matcher . MaxClients {
2017-06-11 07:59:03 +02:00
return false
}
}
return true
}
2017-10-29 08:59:56 +01:00
var (
2021-09-10 03:22:01 +02:00
infoString1 = strings . Split ( `
__ __ ______ ___ ______ ___
__ / // /_/ ____/ __ \/ ____/ __ \
/ _ // __/ __/ / /_/ / / __/ / / /
/ _ // __/ /___/ _, _/ /_/ / /_/ /
/ _ //_/ /_____/_/ |_|\____/\____/
https : //ergo.chat/
https : //github.com/ergochat/ergo
` , "\n" ) [ 1 : ] // XXX: cut off initial blank line
2018-01-23 06:06:55 +01:00
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" )
2020-02-26 08:39:37 +01:00
infoString3 = strings . Split ( ` Jeremy Latt , jlatt
Edmund Huber , edmund - huber
2017-10-29 08:59:56 +01:00
` , "\n" )
)
2022-12-11 06:53:12 +01:00
func ( server * Server ) dumpStacks ( ) {
if gprof := pprof . Lookup ( "goroutine" ) ; gprof != nil {
gprof . WriteTo ( os . Stderr , 2 )
} else {
server . logger . Error ( "internal" , "unable to dump goroutine stacks" )
}
}