package xmpp import ( "crypto/tls" "crypto/x509" "log" "encoding/xml" ) type SessionConnect struct{} type SessionDisconnect struct{} type SessionShouldDisconnect struct{} type session struct { jid string in, out chan any transport *tls.Conn ed encoderDecoder streams []stream rx chan xml.Token } func StartSession(out chan any, jid string, pwd string) chan any { s := session{} 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() return s.in } func (s *session) run() { defer func() { s.out <- SessionDisconnect{} }() err := s.startTransport() if err != nil { return } defer s.transport.Close() s.ed = newEncoderDecoder(s) go s.ed.run() defer func() { s.ed.terminator <- true }() s.openStream() defer s.closeStreams() s.out <- SessionConnect{} for { select { case data := <-s.in: switch data.(type) { case SessionShouldDisconnect: return default: log.Printf("Unknown data '%d'!\n", data) } case _ = <-s.rx: // TODO route received XML token here } } } func (s *session) startTransport() error { domain := domainpart(s.jid) roots, err := x509.SystemCertPool() if err != nil { log.Println(err) return err } s.transport, err = tls.Dial("tcp", domain+":"+"5223", &tls.Config{RootCAs: roots}) if err != nil { log.Println(err) return err } return nil } func (s *session) openStream() { stream := stream{} stream.session = s start := xml.StartElement{ xml.Name{"jabber:client", "stream:stream"}, []xml.Attr{ xml.Attr{xml.Name{"", "from"}, s.jid}, xml.Attr{xml.Name{"", "to"}, domainpart(s.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"}, }, } stream.end = 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] } }