Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

irc.go 2.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. package main
  2. import (
  3. "crypto/tls"
  4. "fmt"
  5. "github.com/spf13/viper"
  6. "github.com/thoj/go-ircevent"
  7. "strings"
  8. "time"
  9. )
  10. func (i *IRCCat) connectIRC(debug bool) error {
  11. irccon := irc.IRC(viper.GetString("irc.nick"), viper.GetString("irc.realname"))
  12. i.irc = irccon
  13. irccon.Debug = debug
  14. irccon.Timeout = time.Second * 15
  15. irccon.RequestCaps = []string{"away-notify", "account-notify", "draft/message-tags-0.2"}
  16. irccon.UseTLS = viper.GetBool("irc.tls")
  17. if viper.IsSet("irc.sasl_pass") && viper.GetString("irc.sasl_pass") != "" {
  18. if viper.IsSet("irc.sasl_login") && viper.GetString("irc.sasl_login") != "" {
  19. irccon.SASLLogin = viper.GetString("irc.sasl_login")
  20. } else {
  21. irccon.SASLLogin = viper.GetString("irc.nick")
  22. }
  23. irccon.SASLPassword = viper.GetString("irc.sasl_pass")
  24. irccon.UseSASL = true
  25. }
  26. if viper.GetBool("irc.sasl_external") {
  27. irccon.SASLMech = "EXTERNAL"
  28. irccon.UseSASL = true
  29. }
  30. if viper.GetBool("irc.tls_skip_verify") {
  31. irccon.TLSConfig = &tls.Config{InsecureSkipVerify: true}
  32. }
  33. if err := addClientCert(irccon); err != nil {
  34. return err
  35. }
  36. irccon.Password = viper.GetString("irc.server_pass")
  37. if err := irccon.Connect(viper.GetString("irc.server")); err != nil {
  38. return err
  39. }
  40. irccon.AddCallback("001", i.handleWelcome)
  41. irccon.AddCallback("PRIVMSG", func(event *irc.Event) {
  42. msg := event.Message()
  43. if (msg[0] == '?' || msg[0] == '!') && len(msg) > 1 {
  44. go i.handleCommand(event)
  45. }
  46. })
  47. irccon.AddCallback("353", i.handleNames)
  48. irccon.AddCallback("JOIN", i.handleJoin)
  49. irccon.AddCallback("PART", i.handlePart)
  50. irccon.AddCallback("QUIT", i.handleQuit)
  51. irccon.AddCallback("KILL", i.handleQuit)
  52. irccon.AddCallback("NICK", i.handleNick)
  53. return nil
  54. }
  55. func (i *IRCCat) handleWelcome(e *irc.Event) {
  56. log.Infof("Negotiated IRCv3 capabilities: %v", i.irc.AcknowledgedCaps)
  57. if viper.IsSet("irc.identify_pass") && viper.GetString("irc.identify_pass") != "" {
  58. i.irc.SendRawf("NICKSERV IDENTIFY %s", viper.GetString("irc.identify_pass"))
  59. }
  60. log.Infof("Connected, joining channels...")
  61. for _, channel := range viper.GetStringSlice("irc.channels") {
  62. key_var := fmt.Sprintf("irc.keys.%s", channel)
  63. if strings.ContainsAny(channel, " \t") {
  64. log.Errorf("Channel name '%s' contains whitespace. Set a channel key by setting the config variable irc.keys.#channel",
  65. channel)
  66. continue
  67. }
  68. if viper.IsSet(key_var) {
  69. i.irc.Join(channel + " " + viper.GetString(key_var))
  70. } else {
  71. i.irc.Join(channel)
  72. }
  73. i.channels.Add(channel)
  74. }
  75. }
  76. func addClientCert(irccon *irc.Connection) error {
  77. certFile := viper.GetString("irc.tls_client_cert") // "" when unset
  78. if certFile == "" {
  79. return nil
  80. }
  81. keyFile := certFile
  82. if k := viper.GetString("irc.tls_client_key"); k != "" {
  83. keyFile = k
  84. }
  85. cert, err := tls.LoadX509KeyPair(certFile, keyFile)
  86. if err != nil {
  87. return err
  88. }
  89. existing := irccon.TLSConfig.Certificates
  90. irccon.TLSConfig.Certificates = append(existing, cert)
  91. return nil
  92. }