package xmpp // import "fluux.io/xmpp" import ( "encoding/xml" "fmt" "reflect" "fluux.io/xmpp/iot" ) /* TODO I would like to be able to write NewIQ(Id, From, To, Type, Lang).AddPayload(IQPayload) Payload would be: payload := Node{ Ns: "http://jabber.org/protocol/disco#info", Tag: "identity", Attrs: map[string]string{ "category":"gateway", "type": "skype", "name": "Test Gateway", }, Nodes: []Node{}, } AddPayload(Ns, Tag, Attrs) ex: NewIQ("get", "test@localhost", "admin@localhost", "en") .AddPayload("http://jabber.org/protocol/disco#info", "identity", map[string]string{ "category":"gateway", "type": "skype", "name": "Test Gateway", }) NewNode(Ns, Tag, Attrs) NewNodeWithChildren(Ns, Tag, Attrs, Nodes) Attr { K string V string } xmpp.Elt.DiscoInfo("identity", "gateway", "skype", "Test Gateway") xmppElt.DiscoInfo.identity(" import xmpp/node/discoinfo discoinfo.Identity("gateway", "skype", "Test Gateway") []Attr{{"category", "gateway"} TODO support ability to put Raw payload */ // ============================================================================ // IQ Packet type IQ struct { // Info/Query XMLName xml.Name `xml:"iq"` PacketAttrs Payload []IQPayload `xml:",omitempty"` RawXML string `xml:",innerxml"` // Error clientError } func NewIQ(iqtype, from, to, id, lang string) IQ { return IQ{ XMLName: xml.Name{Local: "iq"}, PacketAttrs: PacketAttrs{ Id: id, From: from, To: to, Type: iqtype, Lang: lang, }, } } func (iq *IQ) AddPayload(payload IQPayload) { iq.Payload = append(iq.Payload, payload) } func (IQ) Name() string { return "iq" } type iqDecoder struct{} var iq iqDecoder func (iqDecoder) decode(p *xml.Decoder, se xml.StartElement) (IQ, error) { var packet IQ err := p.DecodeElement(&packet, &se) return packet, err } // UnmarshalXML implements custom parsing for IQs func (iq *IQ) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { iq.XMLName = start.Name fmt.Println("IQ Name", iq.XMLName) // Extract IQ attributes for _, attr := range start.Attr { if attr.Name.Local == "id" { iq.Id = attr.Value } if attr.Name.Local == "type" { iq.Type = attr.Value } if attr.Name.Local == "to" { iq.To = attr.Value } if attr.Name.Local == "from" { iq.From = attr.Value } if attr.Name.Local == "lang" { iq.Lang = attr.Value } } // decode inner elements level := 0 for { t, err := d.Token() if err != nil { return err } switch tt := t.(type) { case xml.StartElement: level++ if level <= 1 { var elt interface{} payloadType := tt.Name.Space + " " + tt.Name.Local if payloadType := typeRegistry[payloadType]; payloadType != nil { val := reflect.New(payloadType) elt = val.Interface() } else { elt = new(Node) } if iqPl, ok := elt.(IQPayload); ok { err = d.DecodeElement(elt, &tt) if err != nil { return err } iq.Payload = append(iq.Payload, iqPl) } } case xml.EndElement: level-- if tt == start.End() { return nil } } } } // XMPPFormat returns the string representation of the XMPP packet. // TODO: Should I simply rely on xml.Marshal ? func (iq *IQ) XMPPFormat() string { if iq.Payload != nil { var payload []byte var err error if payload, err = xml.Marshal(iq.Payload); err != nil { return fmt.Sprintf(""+ "", iq.To, iq.Type, iq.Id) } return fmt.Sprintf(""+ "%s", iq.To, iq.Type, iq.Id, payload) } return fmt.Sprintf(""+ "%s", iq.To, iq.Type, iq.Id, iq.RawXML) } // ============================================================================ // Generic IQ Payload type IQPayload interface { IsIQPayload() } type Node struct { XMLName xml.Name Attrs []xml.Attr `xml:"-"` // Content []byte `xml:",innerxml"` Nodes []Node `xml:",any"` } type Attr struct { K string V string } func (n *Node) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { // Assign "n.Attrs = start.Attr", without repeating xmlns in attributes: for _, attr := range start.Attr { // Do not repeat xmlns, it is already in XMLName if attr.Name.Local != "xmlns" { n.Attrs = append(n.Attrs, attr) } } type node Node return d.DecodeElement((*node)(n), &start) } func (n *Node) MarshalXML(e *xml.Encoder, start xml.StartElement) (err error) { start.Attr = n.Attrs start.Name = n.XMLName err = e.EncodeToken(start) e.EncodeElement(n.Nodes, xml.StartElement{Name: n.XMLName}) return e.EncodeToken(xml.EndElement{Name: start.Name}) } func (*Node) IsIQPayload() {} // ============================================================================ // Disco type DiscoInfo struct { XMLName xml.Name `xml:"http://jabber.org/protocol/disco#info query"` Identity Identity `xml:"identity"` Features []Feature `xml:"feature"` } func (*DiscoInfo) IsIQPayload() {} type Identity struct { XMLName xml.Name `xml:"identity,omitempty"` Name string `xml:"name,attr,omitempty"` Category string `xml:"category,attr,omitempty"` Type string `xml:"type,attr,omitempty"` } type Feature struct { XMLName xml.Name `xml:"feature"` Var string `xml:"var,attr"` } // ============================================================================ var typeRegistry = make(map[string]reflect.Type) func init() { typeRegistry["http://jabber.org/protocol/disco#info query"] = reflect.TypeOf(DiscoInfo{}) typeRegistry["urn:ietf:params:xml:ns:xmpp-bind bind"] = reflect.TypeOf(BindBind{}) typeRegistry["urn:xmpp:iot:control set"] = reflect.TypeOf(iot.ControlSet{}) }