1 | package irc
|
---|
2 |
|
---|
3 | import (
|
---|
4 | "bufio"
|
---|
5 | "fmt"
|
---|
6 | "io"
|
---|
7 | )
|
---|
8 |
|
---|
9 | // Conn represents a simple IRC client. It embeds an irc.Reader and an
|
---|
10 | // irc.Writer.
|
---|
11 | type Conn struct {
|
---|
12 | *Reader
|
---|
13 | *Writer
|
---|
14 | }
|
---|
15 |
|
---|
16 | // NewConn creates a new Conn
|
---|
17 | func NewConn(rw io.ReadWriter) *Conn {
|
---|
18 | return &Conn{
|
---|
19 | NewReader(rw),
|
---|
20 | NewWriter(rw),
|
---|
21 | }
|
---|
22 | }
|
---|
23 |
|
---|
24 | // Writer is the outgoing side of a connection.
|
---|
25 | type Writer struct {
|
---|
26 | // DebugCallback is called for each outgoing message. The name of this may
|
---|
27 | // not be stable.
|
---|
28 | DebugCallback func(line string)
|
---|
29 |
|
---|
30 | // Internal fields
|
---|
31 | writer io.Writer
|
---|
32 | writeCallback func(w *Writer, line string) error
|
---|
33 | }
|
---|
34 |
|
---|
35 | func defaultWriteCallback(w *Writer, line string) error {
|
---|
36 | _, err := w.writer.Write([]byte(line + "\r\n"))
|
---|
37 | return err
|
---|
38 | }
|
---|
39 |
|
---|
40 | // NewWriter creates an irc.Writer from an io.Writer.
|
---|
41 | func NewWriter(w io.Writer) *Writer {
|
---|
42 | return &Writer{nil, w, defaultWriteCallback}
|
---|
43 | }
|
---|
44 |
|
---|
45 | // Write is a simple function which will write the given line to the
|
---|
46 | // underlying connection.
|
---|
47 | func (w *Writer) Write(line string) error {
|
---|
48 | if w.DebugCallback != nil {
|
---|
49 | w.DebugCallback(line)
|
---|
50 | }
|
---|
51 |
|
---|
52 | return w.writeCallback(w, line)
|
---|
53 | }
|
---|
54 |
|
---|
55 | // Writef is a wrapper around the connection's Write method and
|
---|
56 | // fmt.Sprintf. Simply use it to send a message as you would normally
|
---|
57 | // use fmt.Printf.
|
---|
58 | func (w *Writer) Writef(format string, args ...interface{}) error {
|
---|
59 | return w.Write(fmt.Sprintf(format, args...))
|
---|
60 | }
|
---|
61 |
|
---|
62 | // WriteMessage writes the given message to the stream
|
---|
63 | func (w *Writer) WriteMessage(m *Message) error {
|
---|
64 | return w.Write(m.String())
|
---|
65 | }
|
---|
66 |
|
---|
67 | // Reader is the incoming side of a connection. The data will be
|
---|
68 | // buffered, so do not re-use the io.Reader used to create the
|
---|
69 | // irc.Reader.
|
---|
70 | type Reader struct {
|
---|
71 | // DebugCallback is called for each incoming message. The name of this may
|
---|
72 | // not be stable.
|
---|
73 | DebugCallback func(string)
|
---|
74 |
|
---|
75 | // Internal fields
|
---|
76 | reader *bufio.Reader
|
---|
77 | }
|
---|
78 |
|
---|
79 | // NewReader creates an irc.Reader from an io.Reader. Note that once a reader is
|
---|
80 | // passed into this function, you should no longer use it as it is being used
|
---|
81 | // inside a bufio.Reader so you cannot rely on only the amount of data for a
|
---|
82 | // Message being read when you call ReadMessage.
|
---|
83 | func NewReader(r io.Reader) *Reader {
|
---|
84 | return &Reader{
|
---|
85 | nil,
|
---|
86 | bufio.NewReader(r),
|
---|
87 | }
|
---|
88 | }
|
---|
89 |
|
---|
90 | // ReadMessage returns the next message from the stream or an error.
|
---|
91 | // It ignores empty messages.
|
---|
92 | func (r *Reader) ReadMessage() (msg *Message, err error) {
|
---|
93 | // It's valid for a message to be empty. Clients should ignore these,
|
---|
94 | // so we do to be good citizens.
|
---|
95 | err = ErrZeroLengthMessage
|
---|
96 | for err == ErrZeroLengthMessage {
|
---|
97 | var line string
|
---|
98 | line, err = r.reader.ReadString('\n')
|
---|
99 | if err != nil {
|
---|
100 | return nil, err
|
---|
101 | }
|
---|
102 |
|
---|
103 | if r.DebugCallback != nil {
|
---|
104 | r.DebugCallback(line)
|
---|
105 | }
|
---|
106 |
|
---|
107 | // Parse the message from our line
|
---|
108 | msg, err = ParseMessage(line)
|
---|
109 | }
|
---|
110 | return msg, err
|
---|
111 | }
|
---|