1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
|
package xmpp
import (
"crypto/tls"
"crypto/x509"
"log"
)
type Event uint8
const (
DisconnectEvent Event = iota
ConnectEvent
ShouldDisconnectEvent
)
type Conn struct {
ch chan Event
jid, pwd string
tcp *tls.Conn
enc encoder
}
func NewConn(ch chan Event, jid string, pwd string) *Conn {
c := Conn{
ch: ch,
jid: jid,
pwd: pwd}
return &c
}
func (c *Conn) Connect() error {
var err error
domain := domainpart(c.jid)
roots, err := x509.SystemCertPool()
if err != nil {
log.Println(err)
return err
}
c.tcp, err = tls.Dial("tcp", domain+":"+"5223", &tls.Config{RootCAs: roots})
if err != nil {
log.Println(err)
return err
}
return nil
}
func (c *Conn) Disconnect() {
c.tcp.Close()
}
func (c *Conn) Run() {
err := c.Connect()
if err != nil {
return
}
defer c.Disconnect()
decoder := newDecoder(c.tcp)
go decoder.run()
defer decoder.stop()
c.enc = newEncoder(c.tcp)
defer c.enc.Close()
tr := newTokenRouter()
end := sendStreamStart(&c.enc, c.jid)
defer sendStreamEnd(&c.enc, end)
c.ch <- ConnectEvent
defer func() { c.ch <- DisconnectEvent }()
for {
select {
case ev := <-c.ch:
switch ev {
case ShouldDisconnectEvent:
return
default:
log.Printf("Unknown Event '%d'!\n", ev)
}
case token := <-decoder.data:
err = tr.route(token, c)
if err != nil {
log.Println(err)
return
}
}
}
}
|