goirc/client/line_test.go

215 lines
6.1 KiB
Go

package client
import (
"reflect"
"testing"
"time"
)
func TestLineCopy(t *testing.T) {
l1 := &Line{
Tags: map[string]string{"foo": "bar", "fizz": "buzz"},
Nick: "nick",
Ident: "ident",
Host: "host",
Src: "src",
Cmd: "cmd",
Raw: "raw",
Args: []string{"arg", "text"},
Time: time.Now(),
}
l2 := l1.Copy()
// Ugly. Couldn't be bothered to bust out reflect and actually think.
if l2.Tags == nil || l2.Tags["foo"] != "bar" || l2.Tags["fizz"] != "buzz" ||
l2.Nick != "nick" || l2.Ident != "ident" || l2.Host != "host" ||
l2.Src != "src" || l2.Cmd != "cmd" || l2.Raw != "raw" ||
l2.Args[0] != "arg" || l2.Args[1] != "text" || l2.Time != l1.Time {
t.Errorf("Line not copied correctly")
t.Errorf("l1: %#v\nl2: %#v", l1, l2)
}
// Now, modify l2 and verify l1 not changed
l2.Tags["foo"] = "baz"
l2.Nick = l2.Nick[1:]
l2.Ident = "foo"
l2.Host = ""
l2.Args[0] = l2.Args[0][1:]
l2.Args[1] = "bar"
l2.Time = time.Now()
if l2.Tags == nil || l2.Tags["foo"] != "baz" || l2.Tags["fizz"] != "buzz" ||
l1.Nick != "nick" || l1.Ident != "ident" || l1.Host != "host" ||
l1.Src != "src" || l1.Cmd != "cmd" || l1.Raw != "raw" ||
l1.Args[0] != "arg" || l1.Args[1] != "text" || l1.Time == l2.Time {
t.Errorf("Original modified when copy changed")
t.Errorf("l1: %#v\nl2: %#v", l1, l2)
}
}
func TestLineText(t *testing.T) {
tests := []struct {
in *Line
out string
}{
{&Line{}, ""},
{&Line{Args: []string{"one thing"}}, "one thing"},
{&Line{Args: []string{"one", "two"}}, "two"},
}
for i, test := range tests {
out := test.in.Text()
if out != test.out {
t.Errorf("test %d: expected: '%s', got '%s'", i, test.out, out)
}
}
}
func TestLineTarget(t *testing.T) {
tests := []struct {
in *Line
out string
}{
{&Line{}, ""},
{&Line{Cmd: JOIN, Args: []string{"#foo"}}, "#foo"},
{&Line{Cmd: PART, Args: []string{"#foo", "bye"}}, "#foo"},
{&Line{Cmd: PRIVMSG, Args: []string{"Me", "la"}, Nick: "Them"}, "Them"},
{&Line{Cmd: NOTICE, Args: []string{"Me", "la"}, Nick: "Them"}, "Them"},
{&Line{Cmd: ACTION, Args: []string{"Me", "la"}, Nick: "Them"}, "Them"},
{&Line{Cmd: CTCP, Args: []string{"PING", "Me", "1"}, Nick: "Them"}, "Them"},
{&Line{Cmd: CTCPREPLY, Args: []string{"PONG", "Me", "2"}, Nick: "Them"}, "Them"},
{&Line{Cmd: PRIVMSG, Args: []string{"#foo", "la"}, Nick: "Them"}, "#foo"},
{&Line{Cmd: NOTICE, Args: []string{"&foo", "la"}, Nick: "Them"}, "&foo"},
{&Line{Cmd: ACTION, Args: []string{"!foo", "la"}, Nick: "Them"}, "!foo"},
{&Line{Cmd: CTCP, Args: []string{"PING", "#foo", "1"}, Nick: "Them"}, "#foo"},
{&Line{Cmd: CTCPREPLY, Args: []string{"PONG", "#foo", "2"}, Nick: "Them"}, "#foo"},
}
for i, test := range tests {
out := test.in.Target()
if out != test.out {
t.Errorf("test %d: expected: '%s', got '%s'", i, test.out, out)
}
}
}
func TestLineTags(t *testing.T) {
tests := []struct {
in string
out *Line
}{
{ // Make sure ERROR lines work
"ERROR :Closing Link: example.org (Too many user connections (global))",
&Line{
Nick: "",
Ident: "",
Host: "",
Src: "",
Cmd: ERROR,
Raw: "ERROR :Closing Link: example.org (Too many user connections (global))",
Args: []string{"Closing Link: example.org (Too many user connections (global))"},
},
},
{ // Make sure non-tagged lines work
":nick!ident@host.com PRIVMSG me :Hello",
&Line{
Nick: "nick",
Ident: "ident",
Host: "host.com",
Src: "nick!ident@host.com",
Cmd: PRIVMSG,
Raw: ":nick!ident@host.com PRIVMSG me :Hello",
Args: []string{"me", "Hello"},
},
},
{ // Tags example from the spec
"@aaa=bbb;ccc;example.com/ddd=eee :nick!ident@host.com PRIVMSG me :Hello",
&Line{
Tags: map[string]string{"aaa": "bbb", "ccc": "", "example.com/ddd": "eee"},
Nick: "nick",
Ident: "ident",
Host: "host.com",
Src: "nick!ident@host.com",
Cmd: PRIVMSG,
Raw: "@aaa=bbb;ccc;example.com/ddd=eee :nick!ident@host.com PRIVMSG me :Hello",
Args: []string{"me", "Hello"},
},
},
{ // Test escaped characters
"@\\:=\\:;\\s=\\s;\\r=\\r;\\n=\\n :nick!ident@host.com PRIVMSG me :Hello",
&Line{
Tags: map[string]string{";": ";", " ": " ", "\r": "\r", "\n": "\n"},
Nick: "nick",
Ident: "ident",
Host: "host.com",
Src: "nick!ident@host.com",
Cmd: PRIVMSG,
Raw: "@\\:=\\:;\\s=\\s;\\r=\\r;\\n=\\n :nick!ident@host.com PRIVMSG me :Hello",
Args: []string{"me", "Hello"},
},
},
{ // Skip empty tag
"@a=a; :nick!ident@host.com PRIVMSG me :Hello",
&Line{
Tags: map[string]string{"a": "a"},
Nick: "nick",
Ident: "ident",
Host: "host.com",
Src: "nick!ident@host.com",
Cmd: PRIVMSG,
Raw: "@a=a; :nick!ident@host.com PRIVMSG me :Hello",
Args: []string{"me", "Hello"},
},
},
{ // = in tag
"@a=a=a; :nick!ident@host.com PRIVMSG me :Hello",
&Line{
Tags: map[string]string{"a": "a=a"},
Nick: "nick",
Ident: "ident",
Host: "host.com",
Src: "nick!ident@host.com",
Cmd: PRIVMSG,
Raw: "@a=a=a; :nick!ident@host.com PRIVMSG me :Hello",
Args: []string{"me", "Hello"},
},
},
}
for i, test := range tests {
got := ParseLine(test.in)
if !reflect.DeepEqual(got, test.out) {
t.Errorf("test %d:\nexpected %#v\ngot %#v", i, test.out, got)
}
}
}
func TestParseUserHost(t *testing.T) {
tests := []struct {
in, nick, ident, host string
ok bool
}{
{"", "", "", "", false},
{" ", "", "", "", false},
{"somestring", "", "", "", false},
{" s p ", "", "", "", false},
{"foo!bar", "", "", "", false},
{"foo@baz.com", "", "", "", false},
{"foo!bar@baz.com", "foo", "bar", "baz.com", true},
{" foo!bar@baz.com", "foo", "bar", "baz.com", true},
{" foo!bar@baz.com ", "foo", "bar", "baz.com", true},
}
for i, test := range tests {
nick, ident, host, ok := parseUserHost(test.in)
if test.nick != nick ||
test.ident != ident ||
test.host != host ||
test.ok != ok {
t.Errorf("%d: parseUserHost(%q) = %q, %q, %q, %t; want %q, %q, %q, %t",
i, test.in, nick, ident, host, ok, test.nick, test.ident, test.host, test.ok)
}
}
}