|
@@ -0,0 +1,380 @@
|
|
|
|
+package ircclient
|
|
|
|
+
|
|
|
|
+import (
|
|
|
|
+ "bufio"
|
|
|
|
+ "crypto/ecdsa"
|
|
|
|
+ "crypto/elliptic"
|
|
|
|
+ "crypto/rand"
|
|
|
|
+ "crypto/tls"
|
|
|
|
+ "crypto/x509"
|
|
|
|
+ "crypto/x509/pkix"
|
|
|
|
+ "encoding/base64"
|
|
|
|
+ "encoding/pem"
|
|
|
|
+ "fmt"
|
|
|
|
+ "math/big"
|
|
|
|
+ rnd "math/rand"
|
|
|
|
+ "net"
|
|
|
|
+ "strconv"
|
|
|
|
+ "strings"
|
|
|
|
+ "testing"
|
|
|
|
+ "time"
|
|
|
|
+)
|
|
|
|
+
|
|
|
|
+func setupSocket() (listen net.Listener, addr string) {
|
|
|
|
+ // establish network socket connection to set Comm_handle
|
|
|
|
+ var err error
|
|
|
|
+ var listener net.Listener
|
|
|
|
+ var address string
|
|
|
|
+
|
|
|
|
+ listener, err = net.Listen("tcp", "127.0.0.1:0")
|
|
|
|
+ if err != nil {
|
|
|
|
+ panic(err)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // I only need address for making the connection.
|
|
|
|
+ // Get address of listening socket
|
|
|
|
+ address = listener.Addr().String()
|
|
|
|
+ return listener, address
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func generateKeyPair() (keypair tls.Certificate) {
|
|
|
|
+ // generate test certificate
|
|
|
|
+ priv, _ := ecdsa.GenerateKey(elliptic.P521(), rand.Reader)
|
|
|
|
+ durationBefore, _ := time.ParseDuration("-1h")
|
|
|
|
+ notBefore := time.Now().Add(durationBefore)
|
|
|
|
+ durationAfter, _ := time.ParseDuration("1h")
|
|
|
|
+ notAfter := time.Now().Add(durationAfter)
|
|
|
|
+ serialNumberLimit := new(big.Int).Lsh(big.NewInt(1), 64)
|
|
|
|
+ serialNumber, _ := rand.Int(rand.Reader, serialNumberLimit)
|
|
|
|
+
|
|
|
|
+ template := x509.Certificate{
|
|
|
|
+ SerialNumber: serialNumber,
|
|
|
|
+ Subject: pkix.Name{
|
|
|
|
+ Organization: []string{"Test Certificate"},
|
|
|
|
+ },
|
|
|
|
+ NotBefore: notBefore,
|
|
|
|
+ NotAfter: notAfter,
|
|
|
|
+ KeyUsage: x509.KeyUsageKeyEncipherment | x509.KeyUsageDigitalSignature | x509.KeyUsageCertSign,
|
|
|
|
+ ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth},
|
|
|
|
+ BasicConstraintsValid: true,
|
|
|
|
+ IsCA: true,
|
|
|
|
+ }
|
|
|
|
+ template.IPAddresses = append(template.IPAddresses, net.ParseIP("127.0.0.1"))
|
|
|
|
+ template.IPAddresses = append(template.IPAddresses, net.ParseIP("::"))
|
|
|
|
+
|
|
|
|
+ derBytes, _ := x509.CreateCertificate(rand.Reader, &template, &template, &priv.PublicKey, priv)
|
|
|
|
+
|
|
|
|
+ c := pem.EncodeToMemory(&pem.Block{Type: "CERTIFICATE", Bytes: derBytes})
|
|
|
|
+ b, _ := x509.MarshalECPrivateKey(priv)
|
|
|
|
+ k := pem.EncodeToMemory(&pem.Block{Type: "EC PRIVATE KEY", Bytes: b})
|
|
|
|
+
|
|
|
|
+ listenerKeyPair, _ := tls.X509KeyPair(c, k)
|
|
|
|
+ return listenerKeyPair
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func setupTLSSocket() (listen net.Listener, addr string) {
|
|
|
|
+ // establish network socket connection to set Comm_handle
|
|
|
|
+ var err error
|
|
|
|
+ var listener net.Listener
|
|
|
|
+ var address string
|
|
|
|
+ var tlsconfig tls.Config
|
|
|
|
+ var keypair tls.Certificate = generateKeyPair()
|
|
|
|
+
|
|
|
|
+ tlsconfig.Certificates = make([]tls.Certificate, 0)
|
|
|
|
+ tlsconfig.Certificates = append(tlsconfig.Certificates, keypair)
|
|
|
|
+
|
|
|
|
+ listener, err = tls.Listen("tcp", "127.0.0.1:0", &tlsconfig)
|
|
|
|
+ if err != nil {
|
|
|
|
+ panic(err)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // I only need address for making the connection.
|
|
|
|
+ // Get address of listening socket
|
|
|
|
+ address = listener.Addr().String()
|
|
|
|
+ return listener, address
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func ircWrite(server net.Conn, output string, t *testing.T) {
|
|
|
|
+ t.Logf(">> %s\n", output)
|
|
|
|
+ server.Write([]byte(output + "\r\n"))
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func ircServer(listener net.Listener, t *testing.T, config *IRCConfig) {
|
|
|
|
+ var server net.Conn
|
|
|
|
+ var err error
|
|
|
|
+
|
|
|
|
+ server, err = listener.Accept()
|
|
|
|
+ if err != nil {
|
|
|
|
+ t.Error("Failed to accept connection.")
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ listener.Close()
|
|
|
|
+ var reader *bufio.Reader = bufio.NewReader(server)
|
|
|
|
+
|
|
|
|
+ var output, line, expect string
|
|
|
|
+ var ping int64 = rnd.Int63()
|
|
|
|
+
|
|
|
|
+ output = fmt.Sprintf("PING :%X", ping)
|
|
|
|
+ ircWrite(server, output, t)
|
|
|
|
+ var parts []string
|
|
|
|
+
|
|
|
|
+ var hasNick, hasUser, hasPing, hasPass bool
|
|
|
|
+ var capSASL bool
|
|
|
|
+ var part1 bool
|
|
|
|
+
|
|
|
|
+ for !part1 {
|
|
|
|
+
|
|
|
|
+ line, err = reader.ReadString('\n')
|
|
|
|
+ if err == nil {
|
|
|
|
+ line = strings.Trim(line, "\r\n")
|
|
|
|
+ // process the received line here
|
|
|
|
+ parts = strings.Split(line, " ")
|
|
|
|
+ t.Logf("<< %s", line)
|
|
|
|
+
|
|
|
|
+ switch parts[0] {
|
|
|
|
+ case "CAP":
|
|
|
|
+ if config.UseTLS && config.UseSASL {
|
|
|
|
+ if line == "CAP REQ :sasl" {
|
|
|
|
+ // Acknowledge we support SASL
|
|
|
|
+ ircWrite(server, ":irc.red-green.com CAP * ACK :sasl", t)
|
|
|
|
+ capSASL = true
|
|
|
|
+ }
|
|
|
|
+ if line == "CAP END" {
|
|
|
|
+ capSASL = true
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ case "AUTHENTICATE":
|
|
|
|
+ if capSASL {
|
|
|
|
+ if line == "AUTHENTICATE PLAIN" {
|
|
|
|
+ ircWrite(server, "AUTHENTICATE +", t)
|
|
|
|
+ } else {
|
|
|
|
+ // Process SASL auth message
|
|
|
|
+ var auth64 string = parts[1]
|
|
|
|
+ byteauth, _ := base64.StdEncoding.DecodeString(auth64)
|
|
|
|
+ var auth string = string(byteauth)
|
|
|
|
+ auth = strings.ReplaceAll(auth, "\x00", " ")
|
|
|
|
+ t.Log(auth)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ case "PASS":
|
|
|
|
+ expect = fmt.Sprintf("PASS %s", config.ServerPassword)
|
|
|
|
+ if expect != line {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", line, expect)
|
|
|
|
+ } else {
|
|
|
|
+ hasPass = true
|
|
|
|
+ }
|
|
|
|
+ case "NICK":
|
|
|
|
+ expect = fmt.Sprintf("NICK %s", config.Nick)
|
|
|
|
+ if expect != line {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", line, expect)
|
|
|
|
+ } else {
|
|
|
|
+ hasNick = true
|
|
|
|
+ }
|
|
|
|
+ case "USER":
|
|
|
|
+ // USER meow-bot 0 * :Meooow! bugz is my owner.
|
|
|
|
+ expect = fmt.Sprintf("USER %s 0 * :%s", config.Username, config.Realname)
|
|
|
|
+ if expect != line {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", line, expect)
|
|
|
|
+ } else {
|
|
|
|
+ hasUser = true
|
|
|
|
+ }
|
|
|
|
+ case "PONG":
|
|
|
|
+ expect = fmt.Sprintf("PONG %X", ping)
|
|
|
|
+ if expect != line {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", line, expect)
|
|
|
|
+ } else {
|
|
|
|
+ hasPing = true
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if !part1 {
|
|
|
|
+ if !capSASL && hasNick && hasUser && hasPing && ((config.ServerPassword == "") || hasPass) {
|
|
|
|
+ part1 = true
|
|
|
|
+
|
|
|
|
+ // part 2:
|
|
|
|
+ var line string
|
|
|
|
+ for _, line = range []string{":irc.red-green.com 001 %s :Welcome to the RedGreen IRC Network",
|
|
|
|
+ ":irc.red-green.com 002 %s :Your host is irc.red-green.com, running version UnrealIRCd-5.2.0.1",
|
|
|
|
+ ":irc.red-green.com 375 %s :- irc.red-green.com Message of the Day -",
|
|
|
|
+ ":irc.red-green.com 372 %s :- ",
|
|
|
|
+ ":irc.red-green.com 376 %s :End of /MOTD command.",
|
|
|
|
+ } {
|
|
|
|
+ output = fmt.Sprintf(line, config.Nick)
|
|
|
|
+ ircWrite(server, output, t)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ } else {
|
|
|
|
+ t.Error("Read Error:", err)
|
|
|
|
+ server.Close()
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if !part1 {
|
|
|
|
+ t.Error("Expected to pass part1 (user/nick/pong)")
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // part 2: nickserv/register
|
|
|
|
+ var part2 bool
|
|
|
|
+
|
|
|
|
+ for _, line = range []string{":[email protected] NOTICE %s :This nickname is registered and protected. If it is your",
|
|
|
|
+ ":[email protected] NOTICE %s :nick, type \x02/msg NickServ IDENTIFY \x1fpassword\x1f\x02. Otherwise,"} {
|
|
|
|
+ output = fmt.Sprintf(line, config.Nick)
|
|
|
|
+ ircWrite(server, output, t)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ for !part2 {
|
|
|
|
+
|
|
|
|
+ line, err = reader.ReadString('\n')
|
|
|
|
+ if err == nil {
|
|
|
|
+ line = strings.Trim(line, "\r\n")
|
|
|
|
+ // process the received line here
|
|
|
|
+ parts = strings.Split(line, " ")
|
|
|
|
+ t.Logf("<< %s", line)
|
|
|
|
+
|
|
|
|
+ switch parts[0] {
|
|
|
|
+ case "NS":
|
|
|
|
+ expect = fmt.Sprintf("NS IDENTIFY %s", config.Password)
|
|
|
|
+ if expect != line {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", line, expect)
|
|
|
|
+ }
|
|
|
|
+ // ok, mark the user as registered
|
|
|
|
+ output = fmt.Sprintf(":[email protected] NOTICE %s :Password accepted - you are now recognized.",
|
|
|
|
+ config.Nick)
|
|
|
|
+ ircWrite(server, output, t)
|
|
|
|
+ output = fmt.Sprintf(":NickServ MODE %s :+r", config.Nick)
|
|
|
|
+ ircWrite(server, output, t)
|
|
|
|
+ part2 = true
|
|
|
|
+ }
|
|
|
|
+ } else {
|
|
|
|
+ t.Error("Read Error:", err)
|
|
|
|
+ server.Close()
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if !part2 {
|
|
|
|
+ t.Error("Expected to pass part2 (ns identify/+r)")
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ time.AfterFunc(time.Millisecond*time.Duration(50), func() { server.Close() })
|
|
|
|
+
|
|
|
|
+ t.Log("Ok, Identified...")
|
|
|
|
+
|
|
|
|
+ for {
|
|
|
|
+ line, err = reader.ReadString('\n')
|
|
|
|
+ if err == nil {
|
|
|
|
+ line = strings.Trim(line, "\r\n")
|
|
|
|
+ // process the received line here
|
|
|
|
+ parts = strings.Split(line, " ")
|
|
|
|
+ t.Logf("<< %s", line)
|
|
|
|
+
|
|
|
|
+ } else {
|
|
|
|
+ t.Log("Read Error:", err)
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+func TestConnect(t *testing.T) {
|
|
|
|
+ var config IRCConfig = IRCConfig{Nick: "test",
|
|
|
|
+ Username: "test",
|
|
|
|
+ Realname: "testing",
|
|
|
|
+ Password: "12345",
|
|
|
|
+ ServerPassword: "allow"}
|
|
|
|
+ var listen net.Listener
|
|
|
|
+ var address string
|
|
|
|
+
|
|
|
|
+ listen, address = setupSocket()
|
|
|
|
+ var parts []string = strings.Split(address, ":")
|
|
|
|
+
|
|
|
|
+ config.Hostname = parts[0]
|
|
|
|
+ config.Port, _ = strconv.Atoi(parts[1])
|
|
|
|
+ go ircServer(listen, t, &config)
|
|
|
|
+ var FromIRC chan IRCMsg
|
|
|
|
+
|
|
|
|
+ FromIRC = make(chan IRCMsg)
|
|
|
|
+ config.ReadChannel = FromIRC
|
|
|
|
+
|
|
|
|
+ config.Connect()
|
|
|
|
+ defer config.Close()
|
|
|
|
+
|
|
|
|
+ var Msg IRCMsg
|
|
|
|
+ var motd, identify bool
|
|
|
|
+ for Msg = range FromIRC {
|
|
|
|
+ if Msg.Cmd == "EndMOTD" {
|
|
|
|
+ t.Log("Got EndMOTD")
|
|
|
|
+ motd = true
|
|
|
|
+ }
|
|
|
|
+ if Msg.Cmd == "Identified" {
|
|
|
|
+ t.Log("Identified")
|
|
|
|
+ identify = true
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if !motd {
|
|
|
|
+ t.Error("Missing EndMOTD")
|
|
|
|
+ }
|
|
|
|
+ if !identify {
|
|
|
|
+ t.Error("Missing Identified")
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if config.MyNick != config.Nick {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", config.MyNick, config.Nick)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+}
|
|
|
|
+func TestConnectTLS(t *testing.T) {
|
|
|
|
+ var config IRCConfig = IRCConfig{Nick: "test",
|
|
|
|
+ Username: "test",
|
|
|
|
+ Realname: "testing",
|
|
|
|
+ Password: "12345",
|
|
|
|
+ UseTLS: true,
|
|
|
|
+ UseSASL: true,
|
|
|
|
+ Insecure: true,
|
|
|
|
+ ServerPassword: "allow"}
|
|
|
|
+ var listen net.Listener
|
|
|
|
+ var address string
|
|
|
|
+
|
|
|
|
+ listen, address = setupTLSSocket()
|
|
|
|
+ var parts []string = strings.Split(address, ":")
|
|
|
|
+
|
|
|
|
+ config.Hostname = parts[0]
|
|
|
|
+ config.Port, _ = strconv.Atoi(parts[1])
|
|
|
|
+ go ircServer(listen, t, &config)
|
|
|
|
+ var FromIRC chan IRCMsg
|
|
|
|
+
|
|
|
|
+ FromIRC = make(chan IRCMsg)
|
|
|
|
+ config.ReadChannel = FromIRC
|
|
|
|
+
|
|
|
|
+ config.Connect()
|
|
|
|
+ defer config.Close()
|
|
|
|
+
|
|
|
|
+ var Msg IRCMsg
|
|
|
|
+ var motd, identify bool
|
|
|
|
+ for Msg = range FromIRC {
|
|
|
|
+ if Msg.Cmd == "EndMOTD" {
|
|
|
|
+ t.Log("Got EndMOTD")
|
|
|
|
+ motd = true
|
|
|
|
+ }
|
|
|
|
+ if Msg.Cmd == "Identified" {
|
|
|
|
+ t.Log("Identified")
|
|
|
|
+ identify = true
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if !motd {
|
|
|
|
+ t.Error("Missing EndMOTD")
|
|
|
|
+ }
|
|
|
|
+ if !identify {
|
|
|
|
+ t.Error("Missing Identified")
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if config.MyNick != config.Nick {
|
|
|
|
+ t.Errorf("Got %s, Expected %s", config.MyNick, config.Nick)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+}
|