go-xmpp/message.go

117 lines
2.4 KiB
Go
Raw Normal View History

package xmpp
import (
"encoding/xml"
)
2018-01-13 09:50:17 -08:00
// ============================================================================
// Message Packet
// Message implements RFC 6120 - A.5 Client Namespace (a part)
2018-01-13 09:50:17 -08:00
type Message struct {
XMLName xml.Name `xml:"message"`
Attrs
Subject string `xml:"subject,omitempty"`
Body string `xml:"body,omitempty"`
Thread string `xml:"thread,omitempty"`
Error Err `xml:"error,omitempty"`
Extensions []MsgExtension `xml:",omitempty"`
}
2018-01-13 09:50:17 -08:00
func (Message) Name() string {
return "message"
}
func NewMessage(a Attrs) Message {
2018-01-24 00:38:02 -08:00
return Message{
XMLName: xml.Name{Local: "message"},
Attrs: a,
2018-01-24 00:38:02 -08:00
}
}
2018-01-13 09:50:17 -08:00
type messageDecoder struct{}
var message messageDecoder
2018-01-13 09:50:17 -08:00
func (messageDecoder) decode(p *xml.Decoder, se xml.StartElement) (Message, error) {
var packet Message
err := p.DecodeElement(&packet, &se)
return packet, err
}
2019-06-04 10:10:21 -07:00
// XMPPFormat with all Extensions
2018-01-13 09:50:17 -08:00
func (msg *Message) XMPPFormat() string {
2019-06-04 10:10:21 -07:00
out, err := xml.MarshalIndent(msg, "", "")
if err != nil {
return ""
}
return string(out)
}
// UnmarshalXML implements custom parsing for IQs
func (msg *Message) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error {
msg.XMLName = start.Name
// Extract packet attributes
for _, attr := range start.Attr {
if attr.Name.Local == "id" {
msg.Id = attr.Value
}
if attr.Name.Local == "type" {
msg.Type = StanzaType(attr.Value)
}
if attr.Name.Local == "to" {
msg.To = attr.Value
}
if attr.Name.Local == "from" {
msg.From = attr.Value
}
if attr.Name.Local == "lang" {
msg.Lang = attr.Value
}
}
// decode inner elements
for {
t, err := d.Token()
if err != nil {
return err
}
switch tt := t.(type) {
case xml.StartElement:
if msgExt := TypeRegistry.GetMsgExtension(tt.Name); msgExt != nil {
// Decode message extension
err = d.DecodeElement(msgExt, &tt)
if err != nil {
return err
}
msg.Extensions = append(msg.Extensions, msgExt)
} else {
// Decode standard message sub-elements
var err error
switch tt.Name.Local {
case "body":
err = d.DecodeElement(&msg.Body, &tt)
case "thread":
err = d.DecodeElement(&msg.Thread, &tt)
case "subject":
err = d.DecodeElement(&msg.Subject, &tt)
case "error":
err = d.DecodeElement(&msg.Error, &tt)
}
if err != nil {
return err
}
}
case xml.EndElement:
if tt == start.End() {
return nil
}
}
}
}