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
}
|