|
@@ -38,6 +38,89 @@ type IRCMsg struct {
|
|
|
Msg string
|
|
|
}
|
|
|
|
|
|
+// Strip out the NICK part of the From message.
|
|
|
+// :[email protected] => NickServ
|
|
|
+func IRCNick(from string) string {
|
|
|
+ if from[0] == ':' {
|
|
|
+ from = from[1:]
|
|
|
+ }
|
|
|
+ var pos int = strings.Index(from, "!")
|
|
|
+ if pos != -1 {
|
|
|
+ from = from[:pos]
|
|
|
+ }
|
|
|
+ return from
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+IRCParse - split line into IRCMsg
|
|
|
+
|
|
|
+Everything after " :" is the Msg.
|
|
|
+Everything before " :" is split into MsgParts[].
|
|
|
+
|
|
|
+If >= 3 MsgParts {
|
|
|
+ To = MsgParts[2]
|
|
|
+}
|
|
|
+if >= 2 MsgParts {
|
|
|
+ From = IrcNic(MsgParts[0])
|
|
|
+ Cmd = MsgParts[1]
|
|
|
+} else {
|
|
|
+ Cmd = MsgParts[0]
|
|
|
+}
|
|
|
+
|
|
|
+Messages are of the following:
|
|
|
+
|
|
|
+:irc.red-green.com 001 test :Welcome to IRC
|
|
|
+^From ^Cmd ^Msg
|
|
|
+ ^To
|
|
|
+^0 ^1 ^2 MsgParts[]
|
|
|
+
|
|
|
+PING :1234567890
|
|
|
+^Cmd ^Msg
|
|
|
+^0 MsgParts[]
|
|
|
+
|
|
|
+*/
|
|
|
+func IRCParse(line string) IRCMsg {
|
|
|
+ var pos int = strings.Index(line, " :")
|
|
|
+ var results IRCMsg
|
|
|
+
|
|
|
+ if pos != -1 {
|
|
|
+ // Message is everything after " :"
|
|
|
+ results.Msg = line[pos+2:]
|
|
|
+ line = line[:pos]
|
|
|
+ }
|
|
|
+
|
|
|
+ results.MsgParts = strings.Split(line, " ")
|
|
|
+
|
|
|
+ if len(results.MsgParts) >= 2 {
|
|
|
+ results.From = IRCNick(results.MsgParts[0])
|
|
|
+ results.Cmd = results.MsgParts[1]
|
|
|
+ } else {
|
|
|
+ results.Cmd = results.MsgParts[0]
|
|
|
+ }
|
|
|
+ if len(results.MsgParts) >= 3 {
|
|
|
+ results.To = results.MsgParts[2]
|
|
|
+ }
|
|
|
+ return results
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+func oldIRCParse(line string) []string {
|
|
|
+ var pos int = strings.Index(line, " :")
|
|
|
+ var message string
|
|
|
+
|
|
|
+ if pos != -1 {
|
|
|
+ message = line[pos+2:]
|
|
|
+ line = line[:pos]
|
|
|
+ }
|
|
|
+ var results []string
|
|
|
+ results = strings.Split(line, " ")
|
|
|
+ if message != "" {
|
|
|
+ results = append(results, message)
|
|
|
+ }
|
|
|
+ return results
|
|
|
+}
|
|
|
+*/
|
|
|
+
|
|
|
type IRCWrite struct {
|
|
|
To string
|
|
|
Output string
|
|
@@ -326,33 +409,6 @@ func (Config *IRCConfig) Close() {
|
|
|
Config.wg.Wait()
|
|
|
}
|
|
|
|
|
|
-func IRCParse(line string) []string {
|
|
|
- var pos int = strings.Index(line, " :")
|
|
|
- var message string
|
|
|
-
|
|
|
- if pos != -1 {
|
|
|
- message = line[pos+2:]
|
|
|
- line = line[:pos]
|
|
|
- }
|
|
|
- var results []string
|
|
|
- results = strings.Split(line, " ")
|
|
|
- if message != "" {
|
|
|
- results = append(results, message)
|
|
|
- }
|
|
|
- return results
|
|
|
-}
|
|
|
-
|
|
|
-func IRCNick(from string) string {
|
|
|
- if from[0] == ':' {
|
|
|
- from = from[1:]
|
|
|
- }
|
|
|
- var pos int = strings.Index(from, "!")
|
|
|
- if pos != -1 {
|
|
|
- from = from[:pos]
|
|
|
- }
|
|
|
- return from
|
|
|
-}
|
|
|
-
|
|
|
func RandomNick(nick string) string {
|
|
|
var result string = nick + "-"
|
|
|
result += strconv.Itoa(rand.Intn(1000))
|
|
@@ -387,7 +443,7 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
for {
|
|
|
var line string
|
|
|
var err error
|
|
|
- var results []string
|
|
|
+ var results IRCMsg
|
|
|
|
|
|
line, err = Config.Reader.ReadString('\n')
|
|
|
if err == nil {
|
|
@@ -398,59 +454,73 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
|
|
|
results = IRCParse(line)
|
|
|
|
|
|
- if results[1] == "433" && !registering {
|
|
|
- // Nick already in use!
|
|
|
- var newNick string = RandomNick(Config.Nick)
|
|
|
- Config.MyNick = newNick
|
|
|
- Config.PriorityWrite("NICK " + newNick)
|
|
|
- }
|
|
|
-
|
|
|
- if results[1] == "PRIVMSG" {
|
|
|
- // Is this an action?
|
|
|
- if len(results) >= 3 {
|
|
|
- if (results[3][0] == '\x01') && (results[3][len(results[3])-1] == '\x01') {
|
|
|
- // ACTION
|
|
|
- results[1] = "CTCP"
|
|
|
- results[3] = results[3][1 : len(results[3])-1]
|
|
|
- log.Println("CTCP:", results[3])
|
|
|
-
|
|
|
- // Process CTCP commands
|
|
|
- if strings.HasPrefix(results[3], "ACTION ") {
|
|
|
- results[1] = "ACTION"
|
|
|
- results[3] = results[3][7:]
|
|
|
- }
|
|
|
+ switch results.Cmd {
|
|
|
+ case "PING":
|
|
|
+ // Ping from Server
|
|
|
+ Config.PriorityWrite("PONG " + results.Msg)
|
|
|
+ case "005":
|
|
|
+ var support string
|
|
|
+ for _, support = range results.MsgParts[3:] {
|
|
|
+ if strings.Contains(support, "=") {
|
|
|
+ var suppart []string = strings.Split(support, "=")
|
|
|
+ Config.ISupport[suppart[0]] = suppart[1]
|
|
|
+ } else {
|
|
|
+ Config.ISupport[support] = ""
|
|
|
+ }
|
|
|
+ }
|
|
|
+ case "433":
|
|
|
+ if !registering {
|
|
|
+ // Nick already in use!
|
|
|
+ var newNick string = RandomNick(Config.Nick)
|
|
|
+ Config.MyNick = newNick
|
|
|
+ Config.PriorityWrite("NICK " + newNick)
|
|
|
+ }
|
|
|
+ case "PRIVMSG":
|
|
|
+ if (results.Msg[0] == '\x01') && (results.Msg[len(results.Msg)-1] == '\x01') {
|
|
|
+ // ACTION
|
|
|
+ results.Cmd = "CTCP"
|
|
|
+ results.Msg = results.Msg[1 : len(results.Msg)-1]
|
|
|
+ if Config.Debug_Output {
|
|
|
+ log.Println("CTCP:", results.Msg)
|
|
|
+ }
|
|
|
|
|
|
- if strings.HasPrefix(results[3], "PING ") {
|
|
|
- Config.WriteTo(IRCNick(results[0]),
|
|
|
- fmt.Sprintf("NOTICE %s :\x01PING %s\x01",
|
|
|
- IRCNick(results[0]),
|
|
|
- results[3][5:]))
|
|
|
- }
|
|
|
+ // Process CTCP commands
|
|
|
+ if strings.HasPrefix(results.Msg, "ACTION ") {
|
|
|
+ results.Cmd = "ACTION"
|
|
|
+ results.Msg = results.Msg[7:]
|
|
|
+ }
|
|
|
|
|
|
- if results[3] == "VERSION" {
|
|
|
- // Send version reply
|
|
|
- var version string
|
|
|
- if Config.Version != "" {
|
|
|
- version = Config.Version + " " + VERSION
|
|
|
- } else {
|
|
|
- version = VERSION
|
|
|
- }
|
|
|
- Config.WriteTo(IRCNick(results[0]),
|
|
|
- fmt.Sprintf("NOTICE %s :\x01VERSION %s\x01",
|
|
|
- IRCNick(results[0]), version))
|
|
|
- }
|
|
|
+ if strings.HasPrefix(results.Msg, "PING ") {
|
|
|
+ Config.WriteTo(IRCNick(results.From),
|
|
|
+ fmt.Sprintf("NOTICE %s :\x01PING %s\x01",
|
|
|
+ IRCNick(results.From),
|
|
|
+ results.Msg[5:]))
|
|
|
+ }
|
|
|
|
|
|
- if results[3] == "TIME" {
|
|
|
- // Send time reply
|
|
|
- var now time.Time = time.Now()
|
|
|
- Config.WriteTo(IRCNick(results[0]),
|
|
|
- fmt.Sprintf("NOTICE %s :\x01TIME %s\x01",
|
|
|
- IRCNick(results[0]),
|
|
|
- now.Format(time.ANSIC)))
|
|
|
+ if results.Msg == "VERSION" {
|
|
|
+ // Send version reply
|
|
|
+ var version string
|
|
|
+ if Config.Version != "" {
|
|
|
+ version = Config.Version + " " + VERSION
|
|
|
+ } else {
|
|
|
+ version = VERSION
|
|
|
}
|
|
|
+ Config.WriteTo(IRCNick(results.From),
|
|
|
+ fmt.Sprintf("NOTICE %s :\x01VERSION %s\x01",
|
|
|
+ IRCNick(results.From), version))
|
|
|
+ }
|
|
|
|
|
|
+ if results.Msg == "TIME" {
|
|
|
+ // Send time reply
|
|
|
+ var now time.Time = time.Now()
|
|
|
+ Config.WriteTo(IRCNick(results.From),
|
|
|
+ fmt.Sprintf("NOTICE %s :\x01TIME %s\x01",
|
|
|
+ IRCNick(results.From),
|
|
|
+ now.Format(time.ANSIC)))
|
|
|
}
|
|
|
+
|
|
|
}
|
|
|
+
|
|
|
}
|
|
|
} else {
|
|
|
// This is likely, 2022/04/05 10:11:41 ReadString: EOF
|
|
@@ -461,17 +531,20 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
return
|
|
|
}
|
|
|
|
|
|
- var msg IRCMsg = IRCMsg{MsgParts: results}
|
|
|
- if len(results) >= 3 {
|
|
|
- msg.From = IRCNick(results[0])
|
|
|
- msg.Cmd = results[1]
|
|
|
- msg.To = results[2]
|
|
|
- if len(results) >= 4 {
|
|
|
- msg.Msg = results[3]
|
|
|
+ var msg IRCMsg = results
|
|
|
+ /*
|
|
|
+ IRCMsg{MsgParts: results}
|
|
|
+ if len(results) >= 3 {
|
|
|
+ msg.From = IRCNick(results[0])
|
|
|
+ msg.Cmd = results[1]
|
|
|
+ msg.To = results[2]
|
|
|
+ if len(results) >= 4 {
|
|
|
+ msg.Msg = results[3]
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ msg.Cmd = results[0]
|
|
|
}
|
|
|
- } else {
|
|
|
- msg.Cmd = results[0]
|
|
|
- }
|
|
|
+ */
|
|
|
|
|
|
if !Config.Debug_Output {
|
|
|
if msg.Cmd == "ERROR" || msg.Cmd[0] == '4' || msg.Cmd[0] == '5' {
|
|
@@ -480,45 +553,28 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- // Answer PING/PONG immediately.
|
|
|
- if results[0] == "PING" {
|
|
|
- Config.PriorityWrite("PONG " + results[1])
|
|
|
- }
|
|
|
-
|
|
|
- if (msg.Cmd == "401") || (msg.Cmd == "404") {
|
|
|
+ if msg.Cmd == "401" || msg.Cmd == "404" {
|
|
|
// No such nick/channel
|
|
|
log.Printf("Remove %s from buffer.", msg.MsgParts[3])
|
|
|
Config.DelChannel <- msg.MsgParts[3]
|
|
|
}
|
|
|
|
|
|
- if msg.Cmd == "005" {
|
|
|
- // ISUPPORT msg.MsgParts[3:len(msg.MsgParts)-1]
|
|
|
- var support string
|
|
|
- for _, support = range msg.MsgParts[3 : len(msg.MsgParts)-1] {
|
|
|
- if strings.Contains(support, "=") {
|
|
|
- var suppart []string = strings.Split(support, "=")
|
|
|
- Config.ISupport[suppart[0]] = suppart[1]
|
|
|
- } else {
|
|
|
- Config.ISupport[support] = ""
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
if !Config.Registered {
|
|
|
// We're not registered yet
|
|
|
|
|
|
// Answer the queries for SASL authentication
|
|
|
- if (msg.Cmd == "CAP") && (msg.Msg == "ACK") {
|
|
|
+ if msg.Cmd == "CAP" && msg.MsgParts[3] == "ACK" {
|
|
|
Config.PriorityWrite("AUTHENTICATE PLAIN")
|
|
|
}
|
|
|
|
|
|
- if msg.Cmd == "CAP" && msg.Msg == "NAK" {
|
|
|
+ if msg.Cmd == "CAP" && msg.MsgParts[3] == "NAK" {
|
|
|
// SASL Authentication failed/not available.
|
|
|
Config.PriorityWrite("CAP END")
|
|
|
Config.UseSASL = false
|
|
|
}
|
|
|
|
|
|
- if (msg.MsgParts[0] == "AUTHENTICATE") && (msg.MsgParts[1] == "+") {
|
|
|
+ // msg.Cmd == "AUTH..."
|
|
|
+ if msg.MsgParts[0] == "AUTHENTICATE" && msg.MsgParts[1] == "+" {
|
|
|
var userpass string = fmt.Sprintf("\x00%s\x00%s", Config.Nick, Config.Password)
|
|
|
var b64 string = base64.StdEncoding.EncodeToString([]byte(userpass))
|
|
|
Config.PriorityWrite("AUTHENTICATE " + b64)
|
|
@@ -571,7 +627,7 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
// Were we kicked, is channel in AutoJoin?
|
|
|
// 2022/04/13 20:02:52 << :[email protected] KICK #bugz meow-bot :bugz
|
|
|
// Msg: ircclient.IRCMsg{MsgParts:[]string{":[email protected]", "KICK", "#bugz", "meow-bot", "bugz"}, From:"bugz", To:"#bugz", Cmd:"KICK", Msg:"meow-bot"}
|
|
|
- if strings.Contains(msg.Msg, Config.MyNick) {
|
|
|
+ if msg.MsgParts[3] == Config.MyNick {
|
|
|
if Config.IsAuto(msg.To) {
|
|
|
// Yes, we were kicked from AutoJoin channel
|
|
|
time.AfterFunc(time.Duration(Config.RejoinDelay)*time.Millisecond, func() { Config.WriteTo(msg.To, "JOIN "+msg.To) })
|
|
@@ -609,8 +665,10 @@ func (Config *IRCConfig) ReaderRoutine() {
|
|
|
// :meow NICK :meow-bot
|
|
|
|
|
|
if msg.From == Config.MyNick {
|
|
|
- Config.SetNick(msg.To)
|
|
|
- log.Println("Nick is now:", Config.MyNick)
|
|
|
+ Config.SetNick(msg.Msg)
|
|
|
+ if Config.Debug_Output {
|
|
|
+ log.Println("Nick is now:", Config.MyNick)
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
}
|