summaryrefslogtreecommitdiff
path: root/xmpp/conn.go
blob: f017889462f66273e55c72cb52e9786b21fb9d2e (plain)
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
			}
		}
	}
}