You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

server.go 44KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535
  1. // Copyright (c) 2012-2014 Jeremy Latt
  2. // Copyright (c) 2014-2015 Edmund Huber
  3. // Copyright (c) 2016- Daniel Oaks <daniel@danieloaks.net>
  4. // released under the MIT license
  5. package irc
  6. import (
  7. "bufio"
  8. "crypto/tls"
  9. "encoding/base64"
  10. "fmt"
  11. "log"
  12. "net"
  13. "net/http"
  14. "os"
  15. "os/signal"
  16. "strconv"
  17. "strings"
  18. "sync"
  19. "syscall"
  20. "time"
  21. "github.com/DanielOaks/girc-go/ircmsg"
  22. "github.com/tidwall/buntdb"
  23. )
  24. var (
  25. // cached because this may be used lots
  26. tooManyClientsMsg = ircmsg.MakeMessage(nil, "", "ERROR", "Too many clients from your IP or network")
  27. tooManyClientsBytes, _ = tooManyClientsMsg.Line()
  28. )
  29. // Limits holds the maximum limits for various things such as topic lengths
  30. type Limits struct {
  31. AwayLen int
  32. ChannelLen int
  33. KickLen int
  34. MonitorEntries int
  35. NickLen int
  36. TopicLen int
  37. ChanListModes int
  38. }
  39. // ListenerInterface represents an interface for a listener.
  40. type ListenerInterface struct {
  41. Listener net.Listener
  42. Events chan ListenerEvent
  43. }
  44. const (
  45. // DestroyListener instructs the listener to destroy itself.
  46. DestroyListener ListenerEventType = iota
  47. // UpdateListener instructs the listener to update itself (grab new certs, etc).
  48. UpdateListener = iota
  49. )
  50. // ListenerEventType is the type of event this is.
  51. type ListenerEventType int
  52. // ListenerEvent is an event that's passed to the listener.
  53. type ListenerEvent struct {
  54. Type ListenerEventType
  55. NewConfig *tls.Config
  56. }
  57. // Server is the main Oragono server.
  58. type Server struct {
  59. accountRegistration *AccountRegistration
  60. accounts map[string]*ClientAccount
  61. authenticationEnabled bool
  62. channels ChannelNameMap
  63. checkIdent bool
  64. clients *ClientLookupSet
  65. commands chan Command
  66. configFilename string
  67. connectionLimits *ConnectionLimits
  68. connectionLimitsMutex sync.Mutex // used when affecting the connection limiter, to make sure rehashing doesn't make things go out-of-whack
  69. ctime time.Time
  70. currentOpers map[*Client]bool
  71. idle chan *Client
  72. isupport *ISupportList
  73. limits Limits
  74. listenerEventActMutex sync.Mutex
  75. listeners map[string]ListenerInterface
  76. listenerUpdateMutex sync.Mutex
  77. monitoring map[string][]Client
  78. motdLines []string
  79. name string
  80. nameCasefolded string
  81. networkName string
  82. newConns chan clientConn
  83. operators map[string]Oper
  84. operclasses map[string]OperClass
  85. password []byte
  86. passwords *PasswordManager
  87. rehashMutex sync.Mutex
  88. rehashSignal chan os.Signal
  89. signals chan os.Signal
  90. store buntdb.DB
  91. whoWas *WhoWasList
  92. }
  93. var (
  94. // ServerExitSignals are the signals the server will exit on.
  95. ServerExitSignals = []os.Signal{
  96. syscall.SIGINT,
  97. syscall.SIGTERM,
  98. syscall.SIGQUIT,
  99. }
  100. )
  101. type clientConn struct {
  102. Conn net.Conn
  103. IsTLS bool
  104. }
  105. // NewServer returns a new Oragono server.
  106. func NewServer(configFilename string, config *Config) *Server {
  107. casefoldedName, err := Casefold(config.Server.Name)
  108. if err != nil {
  109. log.Println(fmt.Sprintf("Server name isn't valid: [%s]", config.Server.Name), err.Error())
  110. return nil
  111. }
  112. // startup check that we have HELP entries for every command
  113. for name := range Commands {
  114. _, exists := Help[strings.ToLower(name)]
  115. if !exists {
  116. log.Fatal("Help entry does not exist for ", name)
  117. }
  118. }
  119. if config.AuthenticationEnabled {
  120. SupportedCapabilities[SASL] = true
  121. }
  122. operClasses, err := config.OperatorClasses()
  123. if err != nil {
  124. log.Fatal("Error loading oper classes:", err.Error())
  125. }
  126. opers, err := config.Operators(operClasses)
  127. if err != nil {
  128. log.Fatal("Error loading operators:", err.Error())
  129. }
  130. connectionLimits, err := NewConnectionLimits(config.Server.ConnectionLimits)
  131. if err != nil {
  132. log.Fatal("Error loading connection limits:", err.Error())
  133. }
  134. server := &Server{
  135. accounts: make(map[string]*ClientAccount),
  136. authenticationEnabled: config.AuthenticationEnabled,
  137. channels: make(ChannelNameMap),
  138. clients: NewClientLookupSet(),
  139. commands: make(chan Command),
  140. configFilename: configFilename,
  141. connectionLimits: connectionLimits,
  142. ctime: time.Now(),
  143. currentOpers: make(map[*Client]bool),
  144. idle: make(chan *Client),
  145. limits: Limits{
  146. AwayLen: int(config.Limits.AwayLen),
  147. ChannelLen: int(config.Limits.ChannelLen),
  148. KickLen: int(config.Limits.KickLen),
  149. MonitorEntries: int(config.Limits.MonitorEntries),
  150. NickLen: int(config.Limits.NickLen),
  151. TopicLen: int(config.Limits.TopicLen),
  152. ChanListModes: int(config.Limits.ChanListModes),
  153. },
  154. listeners: make(map[string]ListenerInterface),
  155. monitoring: make(map[string][]Client),
  156. name: config.Server.Name,
  157. nameCasefolded: casefoldedName,
  158. networkName: config.Network.Name,
  159. newConns: make(chan clientConn),
  160. operclasses: *operClasses,
  161. operators: opers,
  162. signals: make(chan os.Signal, len(ServerExitSignals)),
  163. rehashSignal: make(chan os.Signal, 1),
  164. whoWas: NewWhoWasList(config.Limits.WhowasEntries),
  165. checkIdent: config.Server.CheckIdent,
  166. }
  167. // open data store
  168. db, err := buntdb.Open(config.Datastore.Path)
  169. if err != nil {
  170. log.Fatal(fmt.Sprintf("Failed to open datastore: %s", err.Error()))
  171. }
  172. server.store = *db
  173. // load password manager
  174. err = server.store.View(func(tx *buntdb.Tx) error {
  175. saltString, err := tx.Get(keySalt)
  176. if err != nil {
  177. return fmt.Errorf("Could not retrieve salt string: %s", err.Error())
  178. }
  179. salt, err := base64.StdEncoding.DecodeString(saltString)
  180. if err != nil {
  181. return err
  182. }
  183. pwm := NewPasswordManager(salt)
  184. server.passwords = &pwm
  185. return nil
  186. })
  187. if err != nil {
  188. log.Fatal(fmt.Sprintf("Could not load salt: %s", err.Error()))
  189. }
  190. if config.Server.MOTD != "" {
  191. file, err := os.Open(config.Server.MOTD)
  192. if err == nil {
  193. defer file.Close()
  194. reader := bufio.NewReader(file)
  195. for {
  196. line, err := reader.ReadString('\n')
  197. if err != nil {
  198. break
  199. }
  200. line = strings.TrimRight(line, "\r\n")
  201. // "- " is the required prefix for MOTD, we just add it here to make
  202. // bursting it out to clients easier
  203. line = fmt.Sprintf("- %s", line)
  204. server.motdLines = append(server.motdLines, line)
  205. }
  206. }
  207. }
  208. if config.Server.Password != "" {
  209. server.password = config.Server.PasswordBytes()
  210. }
  211. for _, addr := range config.Server.Listen {
  212. server.createListener(addr, config.TLSListeners())
  213. }
  214. if config.Server.Wslisten != "" {
  215. server.wslisten(config.Server.Wslisten, config.Server.TLSListeners)
  216. }
  217. // registration
  218. accountReg := NewAccountRegistration(config.Registration.Accounts)
  219. server.accountRegistration = &accountReg
  220. // Attempt to clean up when receiving these signals.
  221. signal.Notify(server.signals, ServerExitSignals...)
  222. signal.Notify(server.rehashSignal, syscall.SIGHUP)
  223. server.setISupport()
  224. return server
  225. }
  226. // setISupport sets up our RPL_ISUPPORT reply.
  227. func (server *Server) setISupport() {
  228. maxTargetsString := strconv.Itoa(maxTargets)
  229. // add RPL_ISUPPORT tokens
  230. server.isupport = NewISupportList()
  231. server.isupport.Add("AWAYLEN", strconv.Itoa(server.limits.AwayLen))
  232. server.isupport.Add("CASEMAPPING", "rfc7700")
  233. server.isupport.Add("CHANMODES", strings.Join([]string{ChannelModes{BanMask, ExceptMask, InviteMask}.String(), "", ChannelModes{UserLimit, Key}.String(), ChannelModes{InviteOnly, Moderated, NoOutside, OpOnlyTopic, ChanRoleplaying, Secret}.String()}, ","))
  234. server.isupport.Add("CHANNELLEN", strconv.Itoa(server.limits.ChannelLen))
  235. server.isupport.Add("CHANTYPES", "#")
  236. server.isupport.Add("EXCEPTS", "")
  237. server.isupport.Add("INVEX", "")
  238. server.isupport.Add("KICKLEN", strconv.Itoa(server.limits.KickLen))
  239. server.isupport.Add("MAXLIST", fmt.Sprintf("beI:%s", strconv.Itoa(server.limits.ChanListModes)))
  240. server.isupport.Add("MAXTARGETS", maxTargetsString)
  241. server.isupport.Add("MODES", "")
  242. server.isupport.Add("MONITOR", strconv.Itoa(server.limits.MonitorEntries))
  243. server.isupport.Add("NETWORK", server.networkName)
  244. server.isupport.Add("NICKLEN", strconv.Itoa(server.limits.NickLen))
  245. server.isupport.Add("PREFIX", "(qaohv)~&@%+")
  246. server.isupport.Add("RPCHAN", "E")
  247. server.isupport.Add("RPUSER", "E")
  248. server.isupport.Add("STATUSMSG", "~&@%+")
  249. server.isupport.Add("TARGMAX", fmt.Sprintf("NAMES:1,LIST:1,KICK:1,WHOIS:1,PRIVMSG:%s,NOTICE:%s,MONITOR:", maxTargetsString, maxTargetsString))
  250. server.isupport.Add("TOPICLEN", strconv.Itoa(server.limits.TopicLen))
  251. // account registration
  252. if server.accountRegistration.Enabled {
  253. // 'none' isn't shown in the REGCALLBACKS vars
  254. var enabledCallbacks []string
  255. for _, name := range server.accountRegistration.EnabledCallbacks {
  256. if name != "*" {
  257. enabledCallbacks = append(enabledCallbacks, name)
  258. }
  259. }
  260. server.isupport.Add("REGCOMMANDS", "CREATE,VERIFY")
  261. server.isupport.Add("REGCALLBACKS", strings.Join(enabledCallbacks, ","))
  262. server.isupport.Add("REGCREDTYPES", "passphrase,certfp")
  263. }
  264. server.isupport.RegenerateCachedReply()
  265. }
  266. func loadChannelList(channel *Channel, list string, maskMode ChannelMode) {
  267. if list == "" {
  268. return
  269. }
  270. channel.lists[maskMode].AddAll(strings.Split(list, " "))
  271. }
  272. func (server *Server) Shutdown() {
  273. //TODO(dan): Make sure we disallow new nicks
  274. for _, client := range server.clients.ByNick {
  275. client.Notice("Server is shutting down")
  276. }
  277. if err := server.store.Close(); err != nil {
  278. Log.error.Println("Server.Shutdown store.Close: error:", err)
  279. }
  280. }
  281. // Run starts the server.
  282. func (server *Server) Run() {
  283. // defer closing db/store
  284. defer server.store.Close()
  285. done := false
  286. for !done {
  287. select {
  288. case <-server.signals:
  289. server.Shutdown()
  290. done = true
  291. case <-server.rehashSignal:
  292. // eventually we expect to use HUP to reload config
  293. err := server.rehash()
  294. if err != nil {
  295. Log.error.Println("Failed to rehash:", err.Error())
  296. }
  297. case conn := <-server.newConns:
  298. // check connection limits
  299. ipaddr := net.ParseIP(IPString(conn.Conn.RemoteAddr()))
  300. if ipaddr != nil {
  301. server.connectionLimitsMutex.Lock()
  302. err := server.connectionLimits.AddClient(ipaddr, false)
  303. server.connectionLimitsMutex.Unlock()
  304. if err == nil {
  305. go NewClient(server, conn.Conn, conn.IsTLS)
  306. continue
  307. }
  308. }
  309. // too many connections from one client, tell the client and close the connection
  310. // 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
  311. conn.Conn.Write([]byte(tooManyClientsBytes))
  312. conn.Conn.Close()
  313. case client := <-server.idle:
  314. client.Idle()
  315. }
  316. }
  317. }
  318. //
  319. // IRC protocol listeners
  320. //
  321. // createListener starts the given listeners.
  322. func (server *Server) createListener(addr string, tlsMap map[string]*tls.Config) {
  323. config, listenTLS := tlsMap[addr]
  324. _, alreadyExists := server.listeners[addr]
  325. if alreadyExists {
  326. log.Fatal(server, "listener already exists:", addr)
  327. }
  328. // make listener event channel
  329. listenerEventChannel := make(chan ListenerEvent, 1)
  330. // make listener
  331. listener, err := net.Listen("tcp", addr)
  332. if err != nil {
  333. log.Fatal(server, "listen error: ", err)
  334. }
  335. tlsString := "plaintext"
  336. if listenTLS {
  337. config.ClientAuth = tls.RequestClientCert
  338. listener = tls.NewListener(listener, config)
  339. tlsString = "TLS"
  340. }
  341. // throw our details to the server so we can be modified/killed later
  342. li := ListenerInterface{
  343. Events: listenerEventChannel,
  344. Listener: listener,
  345. }
  346. server.listeners[addr] = li
  347. // start listening
  348. Log.info.Printf("%s listening on %s using %s.", server.name, addr, tlsString)
  349. // setup accept goroutine
  350. go func() {
  351. for {
  352. conn, err := listener.Accept()
  353. if err == nil {
  354. newConn := clientConn{
  355. Conn: conn,
  356. IsTLS: listenTLS,
  357. }
  358. server.newConns <- newConn
  359. }
  360. select {
  361. case event := <-server.listeners[addr].Events:
  362. // this is used to confirm that whoever passed us this event has closed the existing listener correctly (in an attempt to get us to notice the event).
  363. // this is required to keep REHASH from having a very small race possibility of killing the primary listener
  364. server.listenerEventActMutex.Lock()
  365. server.listenerEventActMutex.Unlock()
  366. if event.Type == DestroyListener {
  367. // listener should already be closed, this is just for safety
  368. listener.Close()
  369. return
  370. } else if event.Type == UpdateListener {
  371. // close old listener
  372. listener.Close()
  373. // make new listener
  374. listener, err = net.Listen("tcp", addr)
  375. if err != nil {
  376. log.Fatal(server, "listen error: ", err)
  377. }
  378. tlsString := "plaintext"
  379. if event.NewConfig != nil {
  380. config = event.NewConfig
  381. config.ClientAuth = tls.RequestClientCert
  382. listener = tls.NewListener(listener, config)
  383. tlsString = "TLS"
  384. }
  385. // update server ListenerInterface
  386. li.Listener = listener
  387. server.listenerUpdateMutex.Lock()
  388. server.listeners[addr] = li
  389. server.listenerUpdateMutex.Unlock()
  390. // print notice
  391. Log.info.Printf("%s updated listener %s using %s.", server.name, addr, tlsString)
  392. }
  393. default:
  394. // no events waiting for us, fall-through and continue
  395. }
  396. }
  397. }()
  398. }
  399. //
  400. // websocket listen goroutine
  401. //
  402. func (server *Server) wslisten(addr string, tlsMap map[string]*TLSListenConfig) {
  403. http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
  404. if r.Method != "GET" {
  405. Log.error.Printf("%s method not allowed", server.name)
  406. return
  407. }
  408. // We don't have any subprotocols, so if someone attempts to `new
  409. // WebSocket(server, "subprotocol")` they'll break here, instead of
  410. // getting the default, ambiguous, response from gorilla.
  411. if v, ok := r.Header["Sec-Websocket-Protocol"]; ok {
  412. http.Error(w, fmt.Sprintf("WebSocket subprocotols (e.g. %s) not supported", v), 400)
  413. }
  414. ws, err := upgrader.Upgrade(w, r, nil)
  415. if err != nil {
  416. Log.error.Printf("%s websocket upgrade error: %s", server.name, err)
  417. return
  418. }
  419. newConn := clientConn{
  420. Conn: WSContainer{ws},
  421. IsTLS: false, //TODO(dan): track TLS or not here properly
  422. }
  423. server.newConns <- newConn
  424. })
  425. go func() {
  426. config, listenTLS := tlsMap[addr]
  427. tlsString := "plaintext"
  428. var err error
  429. if listenTLS {
  430. tlsString = "TLS"
  431. }
  432. Log.info.Printf("%s websocket listening on %s using %s.", server.name, addr, tlsString)
  433. if listenTLS {
  434. err = http.ListenAndServeTLS(addr, config.Cert, config.Key, nil)
  435. } else {
  436. err = http.ListenAndServe(addr, nil)
  437. }
  438. if err != nil {
  439. Log.error.Printf("%s listenAndServe (%s) error: %s", server.name, tlsString, err)
  440. }
  441. }()
  442. }
  443. //
  444. // server functionality
  445. //
  446. func (server *Server) tryRegister(c *Client) {
  447. if c.registered || !c.HasNick() || !c.HasUsername() ||
  448. (c.capState == CapNegotiating) {
  449. return
  450. }
  451. c.Register()
  452. // send welcome text
  453. //NOTE(dan): we specifically use the NICK here instead of the nickmask
  454. // see http://modern.ircdocs.horse/#rplwelcome-001 for details on why we avoid using the nickmask
  455. c.Send(nil, server.name, RPL_WELCOME, c.nick, fmt.Sprintf("Welcome to the Internet Relay Network %s", c.nick))
  456. c.Send(nil, server.name, RPL_YOURHOST, c.nick, fmt.Sprintf("Your host is %s, running version %s", server.name, Ver))
  457. c.Send(nil, server.name, RPL_CREATED, c.nick, fmt.Sprintf("This server was created %s", server.ctime.Format(time.RFC1123)))
  458. //TODO(dan): Look at adding last optional [<channel modes with a parameter>] parameter
  459. c.Send(nil, server.name, RPL_MYINFO, c.nick, server.name, Ver, supportedUserModesString, supportedChannelModesString)
  460. c.RplISupport()
  461. server.MOTD(c)
  462. c.Send(nil, c.nickMaskString, RPL_UMODEIS, c.nick, c.ModeString())
  463. }
  464. func (server *Server) MOTD(client *Client) {
  465. if len(server.motdLines) < 1 {
  466. client.Send(nil, server.name, ERR_NOMOTD, client.nick, "MOTD File is missing")
  467. return
  468. }
  469. client.Send(nil, server.name, RPL_MOTDSTART, client.nick, fmt.Sprintf("- %s Message of the day - ", server.name))
  470. for _, line := range server.motdLines {
  471. client.Send(nil, server.name, RPL_MOTD, client.nick, line)
  472. }
  473. client.Send(nil, server.name, RPL_ENDOFMOTD, client.nick, "End of MOTD command")
  474. }
  475. func (server *Server) Id() string {
  476. return server.name
  477. }
  478. func (server *Server) Nick() string {
  479. return server.Id()
  480. }
  481. //
  482. // registration commands
  483. //
  484. // PASS <password>
  485. func passHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  486. if client.registered {
  487. client.Send(nil, server.name, ERR_ALREADYREGISTRED, client.nick, "You may not reregister")
  488. return false
  489. }
  490. // if no password exists, skip checking
  491. if len(server.password) == 0 {
  492. client.authorized = true
  493. return false
  494. }
  495. // check the provided password
  496. password := []byte(msg.Params[0])
  497. if ComparePassword(server.password, password) != nil {
  498. client.Send(nil, server.name, ERR_PASSWDMISMATCH, client.nick, "Password incorrect")
  499. client.Send(nil, server.name, "ERROR", "Password incorrect")
  500. return true
  501. }
  502. client.authorized = true
  503. return false
  504. }
  505. // USER <username> * 0 <realname>
  506. func userHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  507. if client.registered {
  508. client.Send(nil, server.name, ERR_ALREADYREGISTRED, client.nick, "You may not reregister")
  509. return false
  510. }
  511. if !client.authorized {
  512. client.Quit("Bad password")
  513. return true
  514. }
  515. if client.username != "" && client.realname != "" {
  516. return false
  517. }
  518. // confirm that username is valid
  519. //
  520. _, err := CasefoldName(msg.Params[0])
  521. if err != nil {
  522. client.Send(nil, "", "ERROR", "Malformed username")
  523. return true
  524. }
  525. // set user info and log client in
  526. //TODO(dan): Could there be a race condition here with adding/removing the client?
  527. //TODO(dan): we should do something like server.clients.Replace(client) instead
  528. // we do it this way to ONLY replace what hasn't already been set
  529. server.clients.Remove(client)
  530. if !client.HasUsername() {
  531. client.username = "~" + msg.Params[0]
  532. // don't bother updating nickmask here, it's not valid anyway
  533. }
  534. if client.realname == "" {
  535. client.realname = msg.Params[3]
  536. }
  537. server.clients.Add(client)
  538. server.tryRegister(client)
  539. return false
  540. }
  541. // QUIT [<reason>]
  542. func quitHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  543. reason := "Quit"
  544. if len(msg.Params) > 0 {
  545. reason += ": " + msg.Params[0]
  546. }
  547. client.Quit(reason)
  548. return true
  549. }
  550. //
  551. // normal commands
  552. //
  553. // PING <server1> [<server2>]
  554. func pingHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  555. client.Send(nil, server.name, "PONG", msg.Params...)
  556. return false
  557. }
  558. // PONG <server> [ <server2> ]
  559. func pongHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  560. // client gets touched when they send this command, so we don't need to do anything
  561. return false
  562. }
  563. // JOIN <channel>{,<channel>} [<key>{,<key>}]
  564. // JOIN 0
  565. func joinHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  566. // handle JOIN 0
  567. if msg.Params[0] == "0" {
  568. for channel := range client.channels {
  569. channel.Part(client, client.nickCasefolded)
  570. }
  571. return false
  572. }
  573. // handle regular JOINs
  574. channels := strings.Split(msg.Params[0], ",")
  575. var keys []string
  576. if len(msg.Params) > 1 {
  577. keys = strings.Split(msg.Params[1], ",")
  578. }
  579. for i, name := range channels {
  580. casefoldedName, err := CasefoldChannel(name)
  581. if err != nil {
  582. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, name, "No such channel")
  583. continue
  584. }
  585. channel := server.channels.Get(casefoldedName)
  586. if channel == nil {
  587. if len(casefoldedName) > server.limits.ChannelLen {
  588. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, name, "No such channel")
  589. continue
  590. }
  591. channel = NewChannel(server, name, true)
  592. }
  593. var key string
  594. if len(keys) > i {
  595. key = keys[i]
  596. }
  597. channel.Join(client, key)
  598. }
  599. return false
  600. }
  601. // PART <channel>{,<channel>} [<reason>]
  602. func partHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  603. channels := strings.Split(msg.Params[0], ",")
  604. var reason string //TODO(dan): if this isn't supplied here, make sure the param doesn't exist in the PART message sent to other users
  605. if len(msg.Params) > 1 {
  606. reason = msg.Params[1]
  607. }
  608. for _, chname := range channels {
  609. casefoldedChannelName, err := CasefoldChannel(chname)
  610. channel := server.channels.Get(casefoldedChannelName)
  611. if err != nil || channel == nil {
  612. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, chname, "No such channel")
  613. continue
  614. }
  615. channel.Part(client, reason)
  616. }
  617. return false
  618. }
  619. // TOPIC <channel> [<topic>]
  620. func topicHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  621. name, err := CasefoldChannel(msg.Params[0])
  622. channel := server.channels.Get(name)
  623. if err != nil || channel == nil {
  624. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, msg.Params[0], "No such channel")
  625. return false
  626. }
  627. if len(msg.Params) > 1 {
  628. channel.SetTopic(client, msg.Params[1])
  629. } else {
  630. channel.GetTopic(client)
  631. }
  632. return false
  633. }
  634. // PRIVMSG <target>{,<target>} <message>
  635. func privmsgHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  636. clientOnlyTags := GetClientOnlyTags(msg.Tags)
  637. targets := strings.Split(msg.Params[0], ",")
  638. message := msg.Params[1]
  639. for i, targetString := range targets {
  640. // max of four targets per privmsg
  641. if i > maxTargets-1 {
  642. break
  643. }
  644. prefixes, targetString := SplitChannelMembershipPrefixes(targetString)
  645. lowestPrefix := GetLowestChannelModePrefix(prefixes)
  646. target, err := CasefoldChannel(targetString)
  647. if err == nil {
  648. channel := server.channels.Get(target)
  649. if channel == nil {
  650. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, targetString, "No such channel")
  651. continue
  652. }
  653. channel.PrivMsg(lowestPrefix, clientOnlyTags, client, message)
  654. } else {
  655. target, err = CasefoldName(targetString)
  656. user := server.clients.Get(target)
  657. if err != nil || user == nil {
  658. client.Send(nil, server.name, ERR_NOSUCHNICK, target, "No such nick")
  659. continue
  660. }
  661. if !user.capabilities[MessageTags] {
  662. clientOnlyTags = nil
  663. }
  664. user.SendFromClient(client, clientOnlyTags, client.nickMaskString, "PRIVMSG", user.nick, message)
  665. if client.capabilities[EchoMessage] {
  666. client.SendFromClient(client, clientOnlyTags, client.nickMaskString, "PRIVMSG", user.nick, message)
  667. }
  668. if user.flags[Away] {
  669. //TODO(dan): possibly implement cooldown of away notifications to users
  670. client.Send(nil, server.name, RPL_AWAY, user.nick, user.awayMessage)
  671. }
  672. }
  673. }
  674. return false
  675. }
  676. func (client *Client) WhoisChannelsNames(target *Client) []string {
  677. isMultiPrefix := target.capabilities[MultiPrefix]
  678. var chstrs []string
  679. index := 0
  680. for channel := range client.channels {
  681. // channel is secret and the target can't see it
  682. if !target.flags[Operator] && channel.flags[Secret] && !channel.members.Has(target) {
  683. continue
  684. }
  685. chstrs = append(chstrs, channel.members[client].Prefixes(isMultiPrefix)+channel.name)
  686. index++
  687. }
  688. return chstrs
  689. }
  690. // WHOIS [ <target> ] <mask> *( "," <mask> )
  691. func whoisHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  692. var masksString string
  693. //var target string
  694. if len(msg.Params) > 1 {
  695. //target = msg.Params[0]
  696. masksString = msg.Params[1]
  697. } else {
  698. masksString = msg.Params[0]
  699. }
  700. if client.flags[Operator] {
  701. masks := strings.Split(masksString, ",")
  702. for _, mask := range masks {
  703. casefoldedMask, err := Casefold(mask)
  704. if err != nil {
  705. client.Send(nil, client.server.name, ERR_NOSUCHNICK, mask, "No such nick")
  706. continue
  707. }
  708. matches := server.clients.FindAll(casefoldedMask)
  709. if len(matches) == 0 {
  710. client.Send(nil, client.server.name, ERR_NOSUCHNICK, mask, "No such nick")
  711. continue
  712. }
  713. for mclient := range matches {
  714. client.getWhoisOf(mclient)
  715. }
  716. }
  717. } else {
  718. // only get the first request
  719. casefoldedMask, err := Casefold(strings.Split(masksString, ",")[0])
  720. mclient := server.clients.Get(casefoldedMask)
  721. if err != nil || mclient == nil {
  722. client.Send(nil, client.server.name, ERR_NOSUCHNICK, masksString, "No such nick")
  723. // fall through, ENDOFWHOIS is always sent
  724. } else {
  725. client.getWhoisOf(mclient)
  726. }
  727. }
  728. client.Send(nil, server.name, RPL_ENDOFWHOIS, client.nick, masksString, "End of /WHOIS list")
  729. return false
  730. }
  731. func (client *Client) getWhoisOf(target *Client) {
  732. client.Send(nil, client.server.name, RPL_WHOISUSER, client.nick, target.nick, target.username, target.hostname, "*", target.realname)
  733. //TODO(dan): ...one channel per reply? really?
  734. for _, line := range client.WhoisChannelsNames(target) {
  735. client.Send(nil, client.server.name, RPL_WHOISCHANNELS, client.nick, target.nick, line)
  736. }
  737. if target.class != nil {
  738. client.Send(nil, client.server.name, RPL_WHOISOPERATOR, client.nick, target.nick, target.whoisLine)
  739. }
  740. if target.certfp != "" && (client.flags[Operator] || client == target) {
  741. client.Send(nil, client.server.name, RPL_WHOISCERTFP, client.nick, target.nick, fmt.Sprintf("has client certificate fingerprint %s", target.certfp))
  742. }
  743. client.Send(nil, client.server.name, RPL_WHOISIDLE, client.nick, target.nick, strconv.FormatUint(target.IdleSeconds(), 10), strconv.FormatInt(target.SignonTime(), 10), "seconds idle, signon time")
  744. }
  745. // <channel> <user> <host> <server> <nick> ( "H" / "G" ) ["*"] [ ( "@" / "+" ) ]
  746. // :<hopcount> <real name>
  747. func (target *Client) RplWhoReply(channel *Channel, client *Client) {
  748. channelName := "*"
  749. flags := ""
  750. if client.flags[Away] {
  751. flags = "G"
  752. } else {
  753. flags = "H"
  754. }
  755. if client.flags[Operator] {
  756. flags += "*"
  757. }
  758. if channel != nil {
  759. flags += channel.members[client].Prefixes(target.capabilities[MultiPrefix])
  760. channelName = channel.name
  761. }
  762. target.Send(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)
  763. }
  764. func whoChannel(client *Client, channel *Channel, friends ClientSet) {
  765. for member := range channel.members {
  766. if !client.flags[Invisible] || friends[client] {
  767. client.RplWhoReply(channel, member)
  768. }
  769. }
  770. }
  771. // WHO [ <mask> [ "o" ] ]
  772. func whoHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  773. friends := client.Friends()
  774. var mask string
  775. if len(msg.Params) > 0 {
  776. casefoldedMask, err := Casefold(msg.Params[0])
  777. if err != nil {
  778. client.Send(nil, server.name, ERR_UNKNOWNERROR, "WHO", "Mask isn't valid")
  779. return false
  780. }
  781. mask = casefoldedMask
  782. }
  783. //TODO(dan): is this used and would I put this param in the Modern doc?
  784. // if not, can we remove it?
  785. //var operatorOnly bool
  786. //if len(msg.Params) > 1 && msg.Params[1] == "o" {
  787. // operatorOnly = true
  788. //}
  789. if mask == "" {
  790. for _, channel := range server.channels {
  791. whoChannel(client, channel, friends)
  792. }
  793. } else if mask[0] == '#' {
  794. // TODO implement wildcard matching
  795. //TODO(dan): ^ only for opers
  796. channel := server.channels.Get(mask)
  797. if channel != nil {
  798. whoChannel(client, channel, friends)
  799. }
  800. } else {
  801. for mclient := range server.clients.FindAll(mask) {
  802. client.RplWhoReply(nil, mclient)
  803. }
  804. }
  805. client.Send(nil, server.name, RPL_ENDOFWHO, client.nick, mask, "End of WHO list")
  806. return false
  807. }
  808. // OPER <name> <password>
  809. func operHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  810. name, err := CasefoldName(msg.Params[0])
  811. if err != nil {
  812. client.Send(nil, server.name, ERR_PASSWDMISMATCH, client.nick, "Password incorrect")
  813. return true
  814. }
  815. hash := server.operators[name].Pass
  816. password := []byte(msg.Params[1])
  817. err = ComparePassword(hash, password)
  818. if (hash == nil) || (err != nil) {
  819. fmt.Println("2", hash)
  820. client.Send(nil, server.name, ERR_PASSWDMISMATCH, client.nick, "Password incorrect")
  821. return true
  822. }
  823. client.flags[Operator] = true
  824. client.operName = name
  825. client.class = server.operators[name].Class
  826. server.currentOpers[client] = true
  827. client.whoisLine = server.operators[name].WhoisLine
  828. // push new vhost if one is set
  829. if len(server.operators[name].Vhost) > 0 {
  830. originalHost := client.nickMaskString
  831. client.vhost = server.operators[name].Vhost
  832. for fClient := range client.Friends(ChgHost) {
  833. fClient.SendFromClient(client, nil, originalHost, "CHGHOST", client.username, client.vhost)
  834. }
  835. client.updateNickMask()
  836. }
  837. client.Send(nil, server.name, RPL_YOUREOPER, client.nick, "You are now an IRC operator")
  838. //TODO(dan): Should this be sent automagically as part of setting the flag/mode?
  839. modech := ModeChanges{&ModeChange{
  840. mode: Operator,
  841. op: Add,
  842. }}
  843. client.Send(nil, server.name, "MODE", client.nick, modech.String())
  844. return false
  845. }
  846. // rehash reloads the config and applies the changes from the config file.
  847. func (server *Server) rehash() error {
  848. // only let one REHASH go on at a time
  849. server.rehashMutex.Lock()
  850. config, err := LoadConfig(server.configFilename)
  851. if err != nil {
  852. return fmt.Errorf("Error rehashing config file: %s", err.Error())
  853. }
  854. // confirm connectionLimits are fine
  855. connectionLimits, err := NewConnectionLimits(config.Server.ConnectionLimits)
  856. if err != nil {
  857. return fmt.Errorf("Error rehashing config file: %s", err.Error())
  858. }
  859. // confirm operator stuff all exists and is fine
  860. operclasses, err := config.OperatorClasses()
  861. if err != nil {
  862. return fmt.Errorf("Error rehashing config file: %s", err.Error())
  863. }
  864. opers, err := config.Operators(operclasses)
  865. if err != nil {
  866. return fmt.Errorf("Error rehashing config file: %s", err.Error())
  867. }
  868. for client := range server.currentOpers {
  869. _, exists := opers[client.operName]
  870. if !exists {
  871. return fmt.Errorf("Oper [%s] no longer exists (used by client [%s])", client.operName, client.nickMaskString)
  872. }
  873. }
  874. // apply new connectionlimits
  875. server.connectionLimitsMutex.Lock()
  876. server.connectionLimits = connectionLimits
  877. for _, client := range server.clients.ByNick {
  878. ipaddr := net.ParseIP(IPString(client.socket.conn.RemoteAddr()))
  879. if ipaddr != nil {
  880. server.connectionLimits.AddClient(ipaddr, true)
  881. }
  882. }
  883. server.connectionLimitsMutex.Unlock()
  884. // setup new and removed caps
  885. addedCaps := make(CapabilitySet)
  886. removedCaps := make(CapabilitySet)
  887. // SASL
  888. if config.AuthenticationEnabled && !server.authenticationEnabled {
  889. // enabling SASL
  890. SupportedCapabilities[SASL] = true
  891. addedCaps[SASL] = true
  892. }
  893. if !config.AuthenticationEnabled && server.authenticationEnabled {
  894. // disabling SASL
  895. SupportedCapabilities[SASL] = false
  896. removedCaps[SASL] = true
  897. }
  898. server.authenticationEnabled = config.AuthenticationEnabled
  899. // burst new and removed caps
  900. var capBurstClients ClientSet
  901. added := make(map[CapVersion]string)
  902. var removed string
  903. if len(addedCaps) > 0 || len(removedCaps) > 0 {
  904. capBurstClients = server.clients.AllWithCaps(CapNotify)
  905. added[Cap301] = addedCaps.String(Cap301)
  906. added[Cap302] = addedCaps.String(Cap302)
  907. // removed never has values
  908. removed = removedCaps.String(Cap301)
  909. }
  910. for sClient := range capBurstClients {
  911. if len(addedCaps) > 0 {
  912. sClient.Send(nil, server.name, "CAP", sClient.nick, "NEW", added[sClient.capVersion])
  913. }
  914. if len(removedCaps) > 0 {
  915. sClient.Send(nil, server.name, "CAP", sClient.nick, "DEL", removed)
  916. }
  917. }
  918. // set server options
  919. server.limits = Limits{
  920. AwayLen: int(config.Limits.AwayLen),
  921. ChannelLen: int(config.Limits.ChannelLen),
  922. KickLen: int(config.Limits.KickLen),
  923. MonitorEntries: int(config.Limits.MonitorEntries),
  924. NickLen: int(config.Limits.NickLen),
  925. TopicLen: int(config.Limits.TopicLen),
  926. ChanListModes: int(config.Limits.ChanListModes),
  927. }
  928. server.operclasses = *operclasses
  929. server.operators = opers
  930. server.checkIdent = config.Server.CheckIdent
  931. // registration
  932. accountReg := NewAccountRegistration(config.Registration.Accounts)
  933. server.accountRegistration = &accountReg
  934. // set RPL_ISUPPORT
  935. oldISupportList := server.isupport
  936. server.setISupport()
  937. newISupportReplies := oldISupportList.GetDifference(server.isupport)
  938. // push new info to all of our clients
  939. for _, sClient := range server.clients.ByNick {
  940. for _, tokenline := range newISupportReplies {
  941. // ugly trickery ahead
  942. sClient.Send(nil, server.name, RPL_ISUPPORT, append([]string{sClient.nick}, tokenline...)...)
  943. }
  944. }
  945. // destroy old listeners
  946. tlsListeners := config.TLSListeners()
  947. for addr := range server.listeners {
  948. var exists bool
  949. for _, newaddr := range config.Server.Listen {
  950. if newaddr == addr {
  951. exists = true
  952. break
  953. }
  954. }
  955. server.listenerEventActMutex.Lock()
  956. if exists {
  957. // update old listener
  958. server.listeners[addr].Events <- ListenerEvent{
  959. Type: UpdateListener,
  960. NewConfig: tlsListeners[addr],
  961. }
  962. } else {
  963. // destroy nonexistent listener
  964. server.listeners[addr].Events <- ListenerEvent{
  965. Type: DestroyListener,
  966. }
  967. }
  968. // force listener to apply the event right away
  969. server.listeners[addr].Listener.Close()
  970. server.listenerEventActMutex.Unlock()
  971. }
  972. for _, newaddr := range config.Server.Listen {
  973. _, exists := server.listeners[newaddr]
  974. if !exists {
  975. // make new listener
  976. server.createListener(newaddr, tlsListeners)
  977. }
  978. }
  979. server.rehashMutex.Unlock()
  980. return nil
  981. }
  982. // REHASH
  983. func rehashHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  984. err := server.rehash()
  985. if err == nil {
  986. client.Send(nil, server.name, RPL_REHASHING, client.nick, "ircd.yaml", "Rehashing")
  987. } else {
  988. client.Send(nil, server.name, ERR_UNKNOWNERROR, client.nick, "REHASH", err.Error())
  989. }
  990. return false
  991. }
  992. // AWAY [<message>]
  993. func awayHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  994. var isAway bool
  995. var text string
  996. if len(msg.Params) > 0 {
  997. isAway = true
  998. text = msg.Params[0]
  999. if len(text) > server.limits.AwayLen {
  1000. text = text[:server.limits.AwayLen]
  1001. }
  1002. }
  1003. if isAway {
  1004. client.flags[Away] = true
  1005. } else {
  1006. delete(client.flags, Away)
  1007. }
  1008. client.awayMessage = text
  1009. var op ModeOp
  1010. if client.flags[Away] {
  1011. op = Add
  1012. client.Send(nil, server.name, RPL_NOWAWAY, client.nick, "You have been marked as being away")
  1013. } else {
  1014. op = Remove
  1015. client.Send(nil, server.name, RPL_UNAWAY, client.nick, "You are no longer marked as being away")
  1016. }
  1017. //TODO(dan): Should this be sent automagically as part of setting the flag/mode?
  1018. modech := ModeChanges{&ModeChange{
  1019. mode: Away,
  1020. op: op,
  1021. }}
  1022. client.Send(nil, server.name, "MODE", client.nick, client.nick, modech.String())
  1023. // dispatch away-notify
  1024. for friend := range client.Friends(AwayNotify) {
  1025. if client.flags[Away] {
  1026. friend.SendFromClient(client, nil, client.nickMaskString, "AWAY", client.awayMessage)
  1027. } else {
  1028. friend.SendFromClient(client, nil, client.nickMaskString, "AWAY")
  1029. }
  1030. }
  1031. return false
  1032. }
  1033. // ISON <nick>{ <nick>}
  1034. func isonHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1035. var nicks = msg.Params
  1036. var err error
  1037. var casefoldedNick string
  1038. ison := make([]string, 0)
  1039. for _, nick := range nicks {
  1040. casefoldedNick, err = CasefoldName(nick)
  1041. if err != nil {
  1042. continue
  1043. }
  1044. if iclient := server.clients.Get(casefoldedNick); iclient != nil {
  1045. ison = append(ison, iclient.nick)
  1046. }
  1047. }
  1048. client.Send(nil, server.name, RPL_ISON, client.nick, strings.Join(nicks, " "))
  1049. return false
  1050. }
  1051. // MOTD [<target>]
  1052. func motdHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1053. //TODO(dan): hook this up when we have multiple servers I guess???
  1054. //var target string
  1055. //if len(msg.Params) > 0 {
  1056. // target = msg.Params[0]
  1057. //}
  1058. server.MOTD(client)
  1059. return false
  1060. }
  1061. // NOTICE <target>{,<target>} <message>
  1062. func noticeHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1063. clientOnlyTags := GetClientOnlyTags(msg.Tags)
  1064. targets := strings.Split(msg.Params[0], ",")
  1065. message := msg.Params[1]
  1066. for i, targetString := range targets {
  1067. // max of four targets per privmsg
  1068. if i > maxTargets-1 {
  1069. break
  1070. }
  1071. prefixes, targetString := SplitChannelMembershipPrefixes(targetString)
  1072. lowestPrefix := GetLowestChannelModePrefix(prefixes)
  1073. target, cerr := CasefoldChannel(targetString)
  1074. if cerr == nil {
  1075. channel := server.channels.Get(target)
  1076. if channel == nil {
  1077. // errors silently ignored with NOTICE as per RFC
  1078. continue
  1079. }
  1080. channel.PrivMsg(lowestPrefix, clientOnlyTags, client, message)
  1081. } else {
  1082. target, err := CasefoldName(targetString)
  1083. if err != nil {
  1084. continue
  1085. }
  1086. user := server.clients.Get(target)
  1087. if user == nil {
  1088. // errors silently ignored with NOTICE as per RFC
  1089. continue
  1090. }
  1091. if !user.capabilities[MessageTags] {
  1092. clientOnlyTags = nil
  1093. }
  1094. user.SendFromClient(client, clientOnlyTags, client.nickMaskString, "NOTICE", user.nick, message)
  1095. if client.capabilities[EchoMessage] {
  1096. client.SendFromClient(client, clientOnlyTags, client.nickMaskString, "NOTICE", user.nick, message)
  1097. }
  1098. }
  1099. }
  1100. return false
  1101. }
  1102. // KICK <channel>{,<channel>} <user>{,<user>} [<comment>]
  1103. func kickHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1104. channels := strings.Split(msg.Params[0], ",")
  1105. users := strings.Split(msg.Params[1], ",")
  1106. if (len(channels) != len(users)) && (len(users) != 1) {
  1107. client.Send(nil, server.name, ERR_NEEDMOREPARAMS, client.nick, "KICK", "Not enough parameters")
  1108. return false
  1109. }
  1110. kicks := make(map[string]string)
  1111. for index, channel := range channels {
  1112. if len(users) == 1 {
  1113. kicks[channel] = users[0]
  1114. } else {
  1115. kicks[channel] = users[index]
  1116. }
  1117. }
  1118. var comment string
  1119. if len(msg.Params) > 2 {
  1120. comment = msg.Params[2]
  1121. }
  1122. for chname, nickname := range kicks {
  1123. casefoldedChname, err := CasefoldChannel(chname)
  1124. channel := server.channels.Get(casefoldedChname)
  1125. if err != nil || channel == nil {
  1126. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, chname, "No such channel")
  1127. continue
  1128. }
  1129. casefoldedNickname, err := CasefoldName(nickname)
  1130. target := server.clients.Get(casefoldedNickname)
  1131. if err != nil || target == nil {
  1132. client.Send(nil, server.name, ERR_NOSUCHNICK, nickname, "No such nick")
  1133. continue
  1134. }
  1135. // make sure client has privs to kick the given user
  1136. //TODO(dan): split this into a separate function that checks if users have privs
  1137. // over other users, useful for things like -aoh as well
  1138. var hasPrivs bool
  1139. for _, mode := range ChannelPrivModes {
  1140. if channel.members[client][mode] {
  1141. hasPrivs = true
  1142. // admins cannot kick other admins
  1143. if mode == ChannelAdmin && channel.members[target][ChannelAdmin] {
  1144. hasPrivs = false
  1145. }
  1146. break
  1147. } else if channel.members[target][mode] {
  1148. break
  1149. }
  1150. }
  1151. if hasPrivs {
  1152. if comment == "" {
  1153. comment = nickname
  1154. }
  1155. channel.Kick(client, target, comment)
  1156. } else {
  1157. client.Send(nil, client.server.name, ERR_CHANOPRIVSNEEDED, chname, "You're not a channel operator")
  1158. }
  1159. }
  1160. return false
  1161. }
  1162. // LIST [<channel>{,<channel>} [<server>]]
  1163. func listHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1164. var channels []string
  1165. if len(msg.Params) > 0 {
  1166. channels = strings.Split(msg.Params[0], ",")
  1167. }
  1168. var target string
  1169. if len(msg.Params) > 1 {
  1170. target = msg.Params[1]
  1171. }
  1172. //TODO(dan): target server when we have multiple servers
  1173. //TODO(dan): we should continue just fine if it's this current server though
  1174. if target != "" {
  1175. client.Send(nil, server.name, ERR_NOSUCHSERVER, client.nick, target, "No such server")
  1176. return false
  1177. }
  1178. if len(channels) == 0 {
  1179. for _, channel := range server.channels {
  1180. if !client.flags[Operator] && channel.flags[Secret] {
  1181. continue
  1182. }
  1183. client.RplList(channel)
  1184. }
  1185. } else {
  1186. // limit regular users to only listing one channel
  1187. if !client.flags[Operator] {
  1188. channels = channels[:1]
  1189. }
  1190. for _, chname := range channels {
  1191. casefoldedChname, err := CasefoldChannel(chname)
  1192. channel := server.channels.Get(casefoldedChname)
  1193. if err != nil || channel == nil || (!client.flags[Operator] && channel.flags[Secret]) {
  1194. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, chname, "No such channel")
  1195. continue
  1196. }
  1197. client.RplList(channel)
  1198. }
  1199. }
  1200. client.Send(nil, server.name, RPL_LISTEND, client.nick, "End of LIST")
  1201. return false
  1202. }
  1203. func (target *Client) RplList(channel *Channel) {
  1204. // get the correct number of channel members
  1205. var memberCount int
  1206. if target.flags[Operator] || channel.members.Has(target) {
  1207. memberCount = len(channel.members)
  1208. } else {
  1209. for member := range channel.members {
  1210. if !member.flags[Invisible] {
  1211. memberCount += 1
  1212. }
  1213. }
  1214. }
  1215. target.Send(nil, target.server.name, RPL_LIST, target.nick, channel.name, string(memberCount), channel.topic)
  1216. }
  1217. // NAMES [<channel>{,<channel>}]
  1218. func namesHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1219. var channels []string
  1220. if len(msg.Params) > 0 {
  1221. channels = strings.Split(msg.Params[0], ",")
  1222. }
  1223. //var target string
  1224. //if len(msg.Params) > 1 {
  1225. // target = msg.Params[1]
  1226. //}
  1227. if len(channels) == 0 {
  1228. for _, channel := range server.channels {
  1229. channel.Names(client)
  1230. }
  1231. return false
  1232. }
  1233. // limit regular users to only listing one channel
  1234. if !client.flags[Operator] {
  1235. channels = channels[:1]
  1236. }
  1237. for _, chname := range channels {
  1238. casefoldedChname, err := CasefoldChannel(chname)
  1239. channel := server.channels.Get(casefoldedChname)
  1240. if err != nil || channel == nil {
  1241. client.Send(nil, server.name, ERR_NOSUCHCHANNEL, client.nick, chname, "No such channel")
  1242. continue
  1243. }
  1244. channel.Names(client)
  1245. }
  1246. return false
  1247. }
  1248. // VERSION [<server>]
  1249. func versionHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1250. var target string
  1251. if len(msg.Params) > 0 {
  1252. target = msg.Params[0]
  1253. }
  1254. casefoldedTarget, err := Casefold(target)
  1255. if target != "" && (err != nil || casefoldedTarget != server.nameCasefolded) {
  1256. client.Send(nil, server.name, ERR_NOSUCHSERVER, client.nick, target, "No such server")
  1257. return false
  1258. }
  1259. client.Send(nil, server.name, RPL_VERSION, client.nick, Ver, server.name)
  1260. client.RplISupport()
  1261. return false
  1262. }
  1263. // INVITE <nickname> <channel>
  1264. func inviteHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1265. nickname := msg.Params[0]
  1266. channelName := msg.Params[1]
  1267. casefoldedNickname, err := CasefoldName(nickname)
  1268. target := server.clients.Get(casefoldedNickname)
  1269. if err != nil || target == nil {
  1270. client.Send(nil, server.name, ERR_NOSUCHNICK, client.nick, nickname, "No such nick")
  1271. return false
  1272. }
  1273. casefoldedChannelName, err := CasefoldChannel(channelName)
  1274. channel := server.channels.Get(casefoldedChannelName)
  1275. if err != nil || channel == nil {
  1276. client.Send(nil, server.name, RPL_INVITING, client.nick, target.nick, channelName)
  1277. target.Send(nil, client.nickMaskString, "INVITE", target.nick, channel.name)
  1278. return true
  1279. }
  1280. channel.Invite(target, client)
  1281. return false
  1282. }
  1283. // TIME [<server>]
  1284. func timeHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1285. var target string
  1286. if len(msg.Params) > 0 {
  1287. target = msg.Params[0]
  1288. }
  1289. casefoldedTarget, err := Casefold(target)
  1290. if (target != "") && err != nil || (casefoldedTarget != server.nameCasefolded) {
  1291. client.Send(nil, server.name, ERR_NOSUCHSERVER, client.nick, target, "No such server")
  1292. return false
  1293. }
  1294. client.Send(nil, server.name, RPL_TIME, client.nick, server.name, time.Now().Format(time.RFC1123))
  1295. return false
  1296. }
  1297. // KILL <nickname> <comment>
  1298. func killHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1299. nickname := msg.Params[0]
  1300. comment := "<no reason supplied>"
  1301. if len(msg.Params) > 1 {
  1302. comment = msg.Params[1]
  1303. }
  1304. casefoldedNickname, err := CasefoldName(nickname)
  1305. target := server.clients.Get(casefoldedNickname)
  1306. if err != nil || target == nil {
  1307. client.Send(nil, client.server.name, ERR_NOSUCHNICK, nickname, "No such nick")
  1308. return false
  1309. }
  1310. quitMsg := fmt.Sprintf("Killed (%s (%s))", client.nick, comment)
  1311. target.Quit(quitMsg)
  1312. target.destroy()
  1313. return false
  1314. }
  1315. // WHOWAS <nickname> [<count> [<server>]]
  1316. func whowasHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
  1317. nicknames := strings.Split(msg.Params[0], ",")
  1318. var count int64
  1319. if len(msg.Params) > 1 {
  1320. count, _ = strconv.ParseInt(msg.Params[1], 10, 64)
  1321. }
  1322. //var target string
  1323. //if len(msg.Params) > 2 {
  1324. // target = msg.Params[2]
  1325. //}
  1326. for _, nickname := range nicknames {
  1327. results := server.whoWas.Find(nickname, count)
  1328. if len(results) == 0 {
  1329. client.Send(nil, server.name, ERR_WASNOSUCHNICK, client.nick, nickname, "There was no such nickname")
  1330. } else {
  1331. for _, whoWas := range results {
  1332. client.Send(nil, server.name, RPL_WHOWASUSER, client.nick, whoWas.nickname, whoWas.username, whoWas.hostname, "*", whoWas.realname)
  1333. }
  1334. }
  1335. client.Send(nil, server.name, RPL_ENDOFWHOWAS, client.nick, nickname, "End of WHOWAS")
  1336. }
  1337. return false
  1338. }