summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--xmpp/conn.go96
-rw-r--r--xmpp/decoder.go57
-rw-r--r--xmpp/encoder.go79
-rw-r--r--xmpp/routing.go104
-rw-r--r--xmpp/sasl.go59
-rw-r--r--xmpp/session.go18
-rw-r--r--xmpp/stream.go42
7 files changed, 5 insertions, 450 deletions
diff --git a/xmpp/conn.go b/xmpp/conn.go
deleted file mode 100644
index f017889..0000000
--- a/xmpp/conn.go
+++ /dev/null
@@ -1,96 +0,0 @@
-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
- }
- }
- }
-}
diff --git a/xmpp/decoder.go b/xmpp/decoder.go
deleted file mode 100644
index 7139b5a..0000000
--- a/xmpp/decoder.go
+++ /dev/null
@@ -1,57 +0,0 @@
-package xmpp
-
-import (
- "crypto/tls"
- "encoding/xml"
- "log"
- "os"
-)
-
-type decoder struct {
- terminator chan bool
- data chan xml.Token
- decoder *xml.Decoder
- debug *xml.Encoder
-}
-
-func newDecoder(conn *tls.Conn) decoder {
- d := decoder{
- make(chan bool),
- make(chan xml.Token),
- xml.NewDecoder(conn),
- xml.NewEncoder(os.Stdout),
- }
-
- d.debug.Indent("S: ", " ")
-
- return d
-}
-
-func (r *decoder) run() {
- for {
- select {
- case <-r.terminator:
- return
- default:
- t, err := r.decoder.Token()
- if t != nil && err == nil {
- err := r.debug.EncodeToken(t)
- if err != nil {
- log.Println("Could not encode received XML token to debug output!")
- }
- err = r.debug.Flush()
- if err != nil {
- log.Println("Could not flush debug output after receive!")
- }
- r.data <- xml.CopyToken(t)
- }
- if err != nil {
- return // FIXME disconnect on parsing error
- }
- }
- }
-}
-
-func (r *decoder) stop() {
- r.terminator <- true
-}
diff --git a/xmpp/encoder.go b/xmpp/encoder.go
deleted file mode 100644
index 434ca68..0000000
--- a/xmpp/encoder.go
+++ /dev/null
@@ -1,79 +0,0 @@
-package xmpp
-
-import (
- "crypto/tls"
- "encoding/xml"
- "os"
-)
-
-type encoder struct {
- tcp *xml.Encoder
- debug *xml.Encoder
-}
-
-func newEncoder(conn *tls.Conn) encoder {
- e := encoder{
- xml.NewEncoder(conn),
- xml.NewEncoder(os.Stdout),
- }
- e.tcp.Indent("", "")
- e.debug.Indent("C: ", " ")
-
- return e
-}
-
-func (e *encoder) encodeNow(t xml.Token) error {
- err := e.encode(t)
- if err != nil {
- return err
- }
-
- err = e.flush()
- if err != nil {
- return err
- }
-
- return nil
-}
-
-func (e *encoder) encode(t xml.Token) error {
- err := e.tcp.EncodeToken(t)
- if err != nil {
- return err
- }
-
- err = e.debug.EncodeToken(t)
- if err != nil {
- return err
- }
-
- return nil
-}
-
-func (e *encoder) flush() error {
- err := e.tcp.Flush()
- if err != nil {
- return err
- }
-
- err = e.debug.Flush()
- if err != nil {
- return err
- }
-
- return nil
-}
-
-func (e *encoder) Close() error {
- err := e.tcp.Close()
- if err != nil {
- return err
- }
-
- err = e.debug.Close()
- if err != nil {
- return err
- }
-
- return nil
-}
diff --git a/xmpp/routing.go b/xmpp/routing.go
deleted file mode 100644
index 9199874..0000000
--- a/xmpp/routing.go
+++ /dev/null
@@ -1,104 +0,0 @@
-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
-}
diff --git a/xmpp/sasl.go b/xmpp/sasl.go
deleted file mode 100644
index 91cbf6b..0000000
--- a/xmpp/sasl.go
+++ /dev/null
@@ -1,59 +0,0 @@
-package xmpp
-
-import (
- "encoding/xml"
- "encoding/base64"
- "errors"
-)
-
-func sendSaslAuth(b []xml.Token, c *Conn) error {
- mechanisms := make([]string, 0)
- for i, v := range(b) {
- switch token := v.(type) {
- case xml.StartElement:
- expected := xml.Name{"urn:ietf:params:xml:ns:xmpp-sasl", "mechanism"}
- if token.Name == expected {
- if i >= (len(b)-1) { continue }
- switch payload := b[i+1].(type) {
- case xml.CharData:
- mechanisms = append(mechanisms, string(payload))
- }
- }
- }
- }
-
- for _, v := range(mechanisms) {
- if v == "PLAIN" {
- start := xml.StartElement{
- xml.Name{"urn:ietf:params:xml:ns:xmpp-sasl", "auth"},
- []xml.Attr{
- xml.Attr{xml.Name{"", "mechanism"}, "PLAIN"},
- },
- }
-
- data := []byte("\x00" + username(c.jid) + "\x00" + c.pwd)
- dst := make([]byte, base64.StdEncoding.EncodedLen(len(data)))
- base64.StdEncoding.Encode(dst, data)
- payload := xml.CharData(dst)
-
- end := start.End()
-
- c.enc.encodeNow(start)
- c.enc.encode(payload)
- c.enc.encodeNow(end)
-
- return nil
- }
- }
-
- return errors.New("No compatible SASL mechanism given")
-}
-
-func onSaslSuccess(b []xml.Token, c *Conn) error {
- sendStreamStart(&c.enc, c.jid)
- return nil
-}
-
-func onSaslFailure(b []xml.Token, c *Conn) error {
- return errors.New("Authentication failed")
-}
diff --git a/xmpp/session.go b/xmpp/session.go
index 2162d66..e928f01 100644
--- a/xmpp/session.go
+++ b/xmpp/session.go
@@ -16,8 +16,8 @@ type session struct {
in, out chan any
transport *tls.Conn
ed encoderDecoder
- streams []stream
rx chan xml.Token
+ streamEnd xml.EndElement
}
func StartSession(out chan any, jid string, pwd string) chan any {
@@ -26,7 +26,6 @@ func StartSession(out chan any, jid string, pwd string) chan any {
s.jid = jid
s.in = make(chan any)
s.out = out
- s.streams = make([]stream, 0)
s.rx = make(chan xml.Token, 0)
go s.run()
@@ -82,9 +81,6 @@ func (s *session) startTransport() error {
}
func (s *session) openStream() {
- stream := stream{}
- stream.session = s
-
start := xml.StartElement{
xml.Name{"jabber:client", "stream:stream"},
[]xml.Attr{
@@ -96,21 +92,17 @@ func (s *session) openStream() {
},
}
- stream.end = start.End()
+ s.streamEnd = start.End()
err := s.ed.encodeToken(start)
if err != nil {
log.Println("Could not encode stream start!")
}
-
- s.streams = append(s.streams, stream)
}
func (s *session) closeStreams() {
- for {
- limit := len(s.streams)
- if limit <= 0 { break }
- s.streams[limit - 1].terminate()
- s.streams = s.streams[:limit - 1]
+ err := s.ed.encodeToken(s.streamEnd)
+ if err != nil {
+ log.Println("Could not encode stream end!")
}
}
diff --git a/xmpp/stream.go b/xmpp/stream.go
deleted file mode 100644
index d0680aa..0000000
--- a/xmpp/stream.go
+++ /dev/null
@@ -1,42 +0,0 @@
-package xmpp
-
-import (
- "encoding/xml"
- "log"
-)
-
-type stream struct {
- session *session
- end xml.EndElement
-}
-
-func (s *stream) terminate() {
- s.session.ed.encodeToken(s.end)
-}
-
-func sendStreamStart(enc *encoder, jid string) xml.EndElement {
- start := xml.StartElement{
- xml.Name{"jabber:client", "stream:stream"},
- []xml.Attr{
- xml.Attr{xml.Name{"", "from"}, jid},
- xml.Attr{xml.Name{"", "to"}, domainpart(jid)},
- xml.Attr{xml.Name{"", "version"}, "1.0"},
- xml.Attr{xml.Name{"", "xml:lang"}, "en"},
- xml.Attr{xml.Name{"", "xmlns:stream"}, "http://etherx.jabber.org/streams"},
- },
- }
-
- err := enc.encodeNow(start)
- if err != nil {
- log.Println("Could not encode stream start!")
- }
-
- return start.End()
-}
-
-func sendStreamEnd(enc *encoder, end xml.EndElement) {
- err := enc.encodeNow(end)
- if err != nil {
- log.Println("Could not encode stream end!")
- }
-}