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 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"}, }, }, {"@a=a; :nick!ident@host.com PRIVMSG me :Hello", nil}, // Bad inputs {"@a=a=a :nick!ident@host.com PRIVMSG me :Hello", nil}, } 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) } } }