mirror of
				https://github.com/FluuxIO/go-xmpp.git
				synced 2025-10-22 12:35:36 -07:00 
			
		
		
		
	Merge pull request #5 from FluuxIO/parsing
Improve IQ parsing and generation
This commit is contained in:
		| @@ -170,7 +170,8 @@ func bind(t *testing.T, c net.Conn, decoder *xml.Decoder) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	switch iq.Payload.(type) { | 	// TODO Check all elements | ||||||
|  | 	switch iq.Payload[0].(type) { | ||||||
| 	case *bindBind: | 	case *bindBind: | ||||||
| 		result := `<iq id='%s' type='result'> | 		result := `<iq id='%s' type='result'> | ||||||
|   <bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'> |   <bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'> | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/xml" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"fluux.io/xmpp" | 	"fluux.io/xmpp" | ||||||
| @@ -20,8 +21,15 @@ func main() { | |||||||
|  |  | ||||||
| 		switch p := packet.(type) { | 		switch p := packet.(type) { | ||||||
| 		case xmpp.IQ: | 		case xmpp.IQ: | ||||||
| 			switch inner := p.Payload.(type) { | 			switch inner := p.Payload[0].(type) { | ||||||
| 			case *xmpp.Node: | 			case *xmpp.Node: | ||||||
|  | 				fmt.Printf("%q\n", inner) | ||||||
|  |  | ||||||
|  | 				data, err := xml.Marshal(inner) | ||||||
|  | 				if err != nil { | ||||||
|  | 					fmt.Println("cannot marshall payload") | ||||||
|  | 				} | ||||||
|  | 				fmt.Println("data=", string(data)) | ||||||
| 				component.processIQ(p.Type, p.Id, p.From, inner) | 				component.processIQ(p.Type, p.Id, p.From, inner) | ||||||
| 			default: | 			default: | ||||||
| 				fmt.Println("default") | 				fmt.Println("default") | ||||||
|   | |||||||
| @@ -63,7 +63,7 @@ func processMessage(client *xmpp.Client, p *mpg123.Player, packet *xmpp.Message) | |||||||
| } | } | ||||||
|  |  | ||||||
| func processIq(client *xmpp.Client, p *mpg123.Player, packet *xmpp.IQ) { | func processIq(client *xmpp.Client, p *mpg123.Player, packet *xmpp.IQ) { | ||||||
| 	switch payload := packet.Payload.(type) { | 	switch payload := packet.Payload[0].(type) { | ||||||
| 	// We support IOT Control IQ | 	// We support IOT Control IQ | ||||||
| 	case *iot.ControlSet: | 	case *iot.ControlSet: | ||||||
| 		var url string | 		var url string | ||||||
| @@ -76,7 +76,7 @@ func processIq(client *xmpp.Client, p *mpg123.Player, packet *xmpp.IQ) { | |||||||
|  |  | ||||||
| 		playSCURL(p, url) | 		playSCURL(p, url) | ||||||
| 		setResponse := new(iot.ControlSetResponse) | 		setResponse := new(iot.ControlSetResponse) | ||||||
| 		reply := xmpp.IQ{PacketAttrs: xmpp.PacketAttrs{To: packet.From, Type: "result", Id: packet.Id}, Payload: setResponse} | 		reply := xmpp.IQ{PacketAttrs: xmpp.PacketAttrs{To: packet.From, Type: "result", Id: packet.Id}, Payload: []xmpp.IQPayload{setResponse}} | ||||||
| 		client.Send(reply.XMPPFormat()) | 		client.Send(reply.XMPPFormat()) | ||||||
| 		// TODO add Soundclound artist / title retrieval | 		// TODO add Soundclound artist / title retrieval | ||||||
| 		sendUserTune(client, "Radiohead", "Spectre") | 		sendUserTune(client, "Radiohead", "Spectre") | ||||||
|   | |||||||
							
								
								
									
										83
									
								
								iq.go
									
									
									
									
									
								
							
							
						
						
									
										83
									
								
								iq.go
									
									
									
									
									
								
							| @@ -7,18 +7,57 @@ import ( | |||||||
| 	"fluux.io/xmpp/iot" | 	"fluux.io/xmpp/iot" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | TODO I would like to be able to write | ||||||
|  |  | ||||||
|  | 	newIQ(Id, From, To, Type, Lang).AddPayload(IQPayload) | ||||||
|  |  | ||||||
|  | 	xmpp.IQ{ | ||||||
|  | 		XMLName: xml.Name{ | ||||||
|  | 			Space: "", | ||||||
|  | 			Local: "", | ||||||
|  | 		}, | ||||||
|  | 		PacketAttrs: xmpp.PacketAttrs{ | ||||||
|  | 			Id:   "", | ||||||
|  | 			From: "", | ||||||
|  | 			To:   "", | ||||||
|  | 			Type: "", | ||||||
|  | 			Lang: "", | ||||||
|  | 		}, | ||||||
|  | 		Payload: nil, | ||||||
|  | 		RawXML:  "", | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | */ | ||||||
|  |  | ||||||
| // ============================================================================ | // ============================================================================ | ||||||
| // IQ Packet | // IQ Packet | ||||||
|  |  | ||||||
| type IQ struct { // Info/Query | type IQ struct { // Info/Query | ||||||
| 	XMLName xml.Name `xml:"iq"` | 	XMLName xml.Name `xml:"iq"` | ||||||
| 	PacketAttrs | 	PacketAttrs | ||||||
| 	Payload IQPayload `xml:",omitempty"` | 	Payload []IQPayload `xml:",omitempty"` | ||||||
| 	RawXML  string    `xml:",innerxml"` | 	RawXML  string      `xml:",innerxml"` | ||||||
| 	// TODO We need to support detecting the IQ namespace / Query packet |  | ||||||
| 	// 	Error   clientError | 	// 	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 { | func (IQ) Name() string { | ||||||
| 	return "iq" | 	return "iq" | ||||||
| } | } | ||||||
| @@ -33,10 +72,6 @@ func (iqDecoder) decode(p *xml.Decoder, se xml.StartElement) (IQ, error) { | |||||||
| 	return packet, err | 	return packet, err | ||||||
| } | } | ||||||
|  |  | ||||||
| type IQPayload interface { |  | ||||||
| 	IsIQPayload() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // UnmarshalXML implements custom parsing for IQs | // UnmarshalXML implements custom parsing for IQs | ||||||
| func (iq *IQ) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { | func (iq *IQ) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { | ||||||
| 	iq.XMLName = start.Name | 	iq.XMLName = start.Name | ||||||
| @@ -83,7 +118,7 @@ func (iq *IQ) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { | |||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
| 				iq.Payload = p | 				iq.Payload = []IQPayload{p} | ||||||
| 				p = nil | 				p = nil | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| @@ -117,12 +152,38 @@ func (iq *IQ) XMPPFormat() string { | |||||||
| } | } | ||||||
|  |  | ||||||
| // ============================================================================ | // ============================================================================ | ||||||
| // Genery IQ Node | // Generic IQ Payload | ||||||
|  |  | ||||||
|  | type IQPayload interface { | ||||||
|  | 	IsIQPayload() | ||||||
|  | } | ||||||
|  |  | ||||||
| type Node struct { | type Node struct { | ||||||
| 	XMLName xml.Name | 	XMLName xml.Name | ||||||
| 	Content []byte `xml:",innerxml"` | 	Attrs   []xml.Attr `xml:"-"` | ||||||
| 	Nodes   []Node `xml:",any"` | 	// Content []byte     `xml:",innerxml"` | ||||||
|  | 	Nodes []Node `xml:",any"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  | 		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() {} | func (*Node) IsIQPayload() {} | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								iq_test.go
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								iq_test.go
									
									
									
									
									
								
							| @@ -27,3 +27,39 @@ func TestUnmarshalIqs(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestGenerateIq(t *testing.T) { | ||||||
|  | 	iq := NewIQ("get", "admin@localhost", "test@localhost", "1", "en") | ||||||
|  | 	payload := Node{ | ||||||
|  | 		XMLName: xml.Name{ | ||||||
|  | 			Space: "http://jabber.org/protocol/disco#info", | ||||||
|  | 			Local: "query", | ||||||
|  | 		}, | ||||||
|  | 		Nodes: []Node{ | ||||||
|  | 			{XMLName: xml.Name{ | ||||||
|  | 				Space: "http://jabber.org/protocol/disco#info", | ||||||
|  | 				Local: "identity", | ||||||
|  | 			}, | ||||||
|  | 				Attrs: []xml.Attr{ | ||||||
|  | 					{Name: xml.Name{Local: "category"}, Value: "gateway"}, | ||||||
|  | 					{Name: xml.Name{Local: "type"}, Value: "skype"}, | ||||||
|  | 					{Name: xml.Name{Local: "name"}, Value: "Test Gateway"}, | ||||||
|  | 				}, | ||||||
|  | 				Nodes: nil, | ||||||
|  | 			}}, | ||||||
|  | 	} | ||||||
|  | 	iq.AddPayload(&payload) | ||||||
|  | 	data, err := xml.Marshal(iq) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Errorf("cannot marshal xml structure") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var parsedIQ = new(IQ) | ||||||
|  | 	if err = xml.Unmarshal(data, parsedIQ); err != nil { | ||||||
|  | 		t.Errorf("Unmarshal(%s) returned error", data) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !reflect.DeepEqual(parsedIQ.Payload[0], iq.Payload[0]) { | ||||||
|  | 		t.Errorf("expecting result %+v = %+v", parsedIQ.Payload[0], iq.Payload[0]) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
| @@ -163,7 +163,8 @@ func (s *Session) bind(o Options) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	switch payload := iq.Payload.(type) { | 	// TODO Check all elements | ||||||
|  | 	switch payload := iq.Payload[0].(type) { | ||||||
| 	case *bindBind: | 	case *bindBind: | ||||||
| 		s.BindJid = payload.Jid // our local id (with possibly randomly generated resource | 		s.BindJid = payload.Jid // our local id (with possibly randomly generated resource | ||||||
| 	default: | 	default: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Mickaël Rémond
					Mickaël Rémond