go-xmpp/component.go

217 lines
6.0 KiB
Go
Raw Normal View History

package xmpp
2018-01-11 13:15:54 -08:00
import (
"context"
"crypto/sha1"
"encoding/hex"
2018-01-11 14:00:59 -08:00
"encoding/xml"
"errors"
2018-01-11 13:15:54 -08:00
"fmt"
"gosrc.io/xmpp/stanza"
"io"
2018-01-11 13:15:54 -08:00
)
type ComponentOptions struct {
2019-10-10 21:24:47 -07:00
TransportConfiguration
// =================================
// Component Connection Info
// Domain is the XMPP server subdomain that the component will handle
Domain string
// Secret is the "password" used by the XMPP server to secure component access
Secret string
// =================================
// Component discovery
// Component human readable name, that will be shown in XMPP discovery
Name string
// Typical categories and types: https://xmpp.org/registrar/disco-categories.html
Category string
Type string
// =================================
// Communication with developer client / StreamManager
// Track and broadcast connection state
EventManager
}
2018-01-11 13:15:54 -08:00
// Component implements an XMPP extension allowing to extend XMPP server
// using external components. Component specifications are defined
// in XEP-0114, XEP-0355 and XEP-0356.
type Component struct {
ComponentOptions
router *Router
2019-10-10 21:24:47 -07:00
transport Transport
2018-01-11 14:00:59 -08:00
// read / write
socketProxy io.ReadWriter // TODO
2018-01-11 13:15:54 -08:00
}
func NewComponent(opts ComponentOptions, r *Router) (*Component, error) {
c := Component{ComponentOptions: opts, router: r}
2019-06-09 03:48:22 -07:00
return &c, nil
}
2018-01-25 14:16:55 -08:00
// Connect triggers component connection to XMPP server component port.
// TODO: Failed handshake should be a permanent error
func (c *Component) Connect() error {
2019-09-05 13:01:40 -07:00
var state SMState
return c.Resume(state)
}
2019-09-05 13:01:40 -07:00
func (c *Component) Resume(sm SMState) error {
2018-01-11 13:15:54 -08:00
var err error
var streamId string
if c.ComponentOptions.TransportConfiguration.Domain == "" {
c.ComponentOptions.TransportConfiguration.Domain = c.ComponentOptions.Domain
2018-01-11 14:00:59 -08:00
}
2019-10-25 06:22:01 -07:00
c.transport, err = NewComponentTransport(c.ComponentOptions.TransportConfiguration)
if err != nil {
c.updateState(StatePermanentError)
return NewConnError(err, true)
2019-10-25 06:22:01 -07:00
}
2018-01-11 14:00:59 -08:00
if streamId, err = c.transport.Connect(); err != nil {
c.updateState(StatePermanentError)
return NewConnError(err, true)
2018-01-11 13:15:54 -08:00
}
c.updateState(StateConnected)
2018-01-11 13:15:54 -08:00
// Authentication
2019-10-10 21:24:47 -07:00
if _, err := fmt.Fprintf(c.transport, "<handshake>%s</handshake>", c.handshake(streamId)); err != nil {
2019-09-05 13:01:40 -07:00
c.updateState(StateStreamError)
return NewConnError(errors.New("cannot send handshake "+err.Error()), false)
2018-01-11 14:00:59 -08:00
}
// Check server response for authentication
val, err := stanza.NextPacket(c.transport.GetDecoder())
2018-01-11 14:00:59 -08:00
if err != nil {
c.updateState(StatePermanentError)
2019-09-05 13:01:40 -07:00
return NewConnError(err, true)
2018-01-11 14:00:59 -08:00
}
switch v := val.(type) {
case stanza.StreamError:
2019-09-05 13:01:40 -07:00
c.streamError("conflict", "no auth loop")
return NewConnError(errors.New("handshake failed "+v.Error.Local), true)
case stanza.Handshake:
// Start the receiver go routine
2019-09-05 13:01:40 -07:00
c.updateState(StateSessionEstablished)
go c.recv()
2018-01-12 09:14:41 -08:00
return nil
2018-01-11 14:00:59 -08:00
default:
c.updateState(StatePermanentError)
2019-09-05 13:01:40 -07:00
return NewConnError(errors.New("expecting handshake result, got "+v.Name()), true)
2018-01-11 14:00:59 -08:00
}
2018-01-12 09:14:41 -08:00
}
2018-01-11 14:00:59 -08:00
func (c *Component) Disconnect() {
// TODO: Add a way to wait for stream close acknowledgement from the server for clean disconnect
2019-10-10 21:24:47 -07:00
if c.transport != nil {
_ = c.transport.Close()
}
}
func (c *Component) SetHandler(handler EventHandler) {
c.Handler = handler
}
// Receiver Go routine receiver
func (c *Component) recv() (err error) {
for {
val, err := stanza.NextPacket(c.transport.GetDecoder())
if err != nil {
c.updateState(StateDisconnected)
return err
}
// Handle stream errors
switch p := val.(type) {
case stanza.StreamError:
c.router.route(c, val)
c.streamError(p.Error.Local, p.Text)
return errors.New("stream error: " + p.Error.Local)
}
c.router.route(c, val)
}
2018-01-12 10:08:47 -08:00
}
2018-01-26 00:55:39 -08:00
// Send marshalls XMPP stanza and sends it to the server.
func (c *Component) Send(packet stanza.Packet) error {
2019-10-10 21:24:47 -07:00
transport := c.transport
if transport == nil {
return errors.New("component is not connected")
}
2018-01-17 09:47:34 -08:00
data, err := xml.Marshal(packet)
if err != nil {
return errors.New("cannot marshal packet " + err.Error())
}
2019-10-10 21:24:47 -07:00
if _, err := fmt.Fprintf(transport, string(data)); err != nil {
2018-01-17 09:47:34 -08:00
return errors.New("cannot send packet " + err.Error())
}
return nil
}
// SendIQ sends an IQ set or get stanza to the server. If a result is received
// the provided handler function will automatically be called.
//
// The provided context should have a timeout to prevent the client from waiting
// forever for an IQ result. For example:
//
// ctx, _ := context.WithTimeout(context.Background(), 30 * time.Second)
// result := <- client.SendIQ(ctx, iq)
//
func (c *Component) SendIQ(ctx context.Context, iq stanza.IQ) (chan stanza.IQ, error) {
if iq.Attrs.Type != "set" && iq.Attrs.Type != "get" {
return nil, ErrCanOnlySendGetOrSetIq
}
if err := c.Send(iq); err != nil {
return nil, err
}
return c.router.NewIQResultRoute(ctx, iq.Attrs.Id), nil
}
2018-01-26 00:55:39 -08:00
// SendRaw sends an XMPP stanza as a string to the server.
// It can be invalid XML or XMPP content. In that case, the server will
// disconnect the component. It is up to the user of this method to
// carefully craft the XML content to produce valid XMPP.
func (c *Component) SendRaw(packet string) error {
2019-10-10 21:24:47 -07:00
transport := c.transport
if transport == nil {
return errors.New("component is not connected")
}
2019-06-09 03:21:20 -07:00
var err error
2019-10-10 21:24:47 -07:00
_, err = fmt.Fprintf(transport, packet)
2019-06-09 03:21:20 -07:00
return err
2018-01-26 00:55:39 -08:00
}
2018-01-25 14:16:55 -08:00
// handshake generates an authentication token based on StreamID and shared secret.
func (c *Component) handshake(streamId string) string {
// 1. Concatenate the Stream ID received from the server with the shared secret.
concatStr := streamId + c.Secret
// 2. Hash the concatenated string according to the SHA1 algorithm, i.e., SHA1( concat (sid, password)).
h := sha1.New()
h.Write([]byte(concatStr))
hash := h.Sum(nil)
// 3. Ensure that the hash output is in hexadecimal format, not binary or base64.
// 4. Convert the hash output to all lowercase characters.
encodedStr := hex.EncodeToString(hash)
return encodedStr
}
2019-06-09 04:02:58 -07:00
/*
TODO: Add support for discovery management directly in component
TODO: Support multiple identities on disco info
TODO: Support returning features on disco info
*/