summaryrefslogtreecommitdiff
path: root/xmpp/routing.go
blob: 919987461dc17fdb1c7719b7fead65fbbf1e788c (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
97
98
99
100
101
102
103
104
package xmpp

import (
	"encoding/xml"
	"errors"
	"log"
)

type tokenRouter struct {
	start        xml.StartElement
	end          xml.EndElement
	buffer       []xml.Token
	level        uint16 // XML nesting level
	enc          *encoder
}

func newTokenRouter() tokenRouter {
	return tokenRouter{
		buffer: make([]xml.Token, 0),
		level:  0,
	}
}

func (r *tokenRouter) route(t xml.Token, c *Conn) error {
	r.buffer = append(r.buffer, t)

	switch unwrapped := t.(type) {
	case xml.StartElement:
		r.level += 1
		if r.level == 1 {
			r.start = unwrapped
			r.buffer = r.buffer[:0]
			// call start handler
		}
	case xml.EndElement:
		if r.level == 0 {
			log.Println("Ignoring XML end element on nesting level zero")
			return nil
		}
		r.level -= 1
		switch r.level {
		case 0:
			return errors.New("Stream was closed by server")
		case 1:
			err := routeElement(r.buffer, c)
			if err != nil {
				return err
			}
			r.buffer = r.buffer[:0]
		}
	case xml.ProcInst:
		log.Println("Ignoring xml.ProcInst element")
		r.buffer = r.buffer[:len(r.buffer)-1]
	case xml.Directive:
		log.Println("Ignoring xml.Directive element")
		r.buffer = r.buffer[:len(r.buffer)-1]
	case xml.Comment:
		log.Println("Ignoring xml.Comment element")
		r.buffer = r.buffer[:len(r.buffer)-1]
	}

	return nil
}

func routeElement(b []xml.Token, c *Conn) error {
	tab := elementRoutingTable{
		{xml.Name{"http://etherx.jabber.org/streams", "features"}, streamFeatures},
		{xml.Name{"urn:ietf:params:xml:ns:xmpp-sasl", "success"}, onSaslSuccess},
		{xml.Name{"urn:ietf:params:xml:ns:xmpp-sasl", "failure"}, onSaslFailure},
	}

	switch unwrapped := b[0].(type) {
	case xml.StartElement:
		for _, v := range(tab) {
			if unwrapped.Name == v.name {
				err := v.handler(b, c)
				if err != nil {
					return err
				}
				return nil
			}
		}
		log.Printf("Ignoring XML element '%s' from namespace '%s'",
		           unwrapped.Name.Local, unwrapped.Name.Space)
	default:
		log.Println("Ignoring XML element which has no StartElement as first token")
	}

	return nil
}

type elementRoutingTable []struct {
	name    xml.Name
	handler func(b []xml.Token, c *Conn) error
}

func streamFeatures(b []xml.Token, c *Conn) error {
	err := sendSaslAuth(b, c)
	if err != nil {
		return err
	}

	return nil
}