From d170626154a4853be386c7180050afe09be3ca0b Mon Sep 17 00:00:00 2001 From: xengineering Date: Wed, 28 Jun 2023 10:56:53 +0200 Subject: xmpp: Implement basic elementBuffer This is needed to buffer XML elements of a stream until they are complete and can be given to an element handler. --- xmpp/element_buffer.go | 58 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 58 insertions(+) create mode 100644 xmpp/element_buffer.go (limited to 'xmpp/element_buffer.go') diff --git a/xmpp/element_buffer.go b/xmpp/element_buffer.go new file mode 100644 index 0000000..c55279b --- /dev/null +++ b/xmpp/element_buffer.go @@ -0,0 +1,58 @@ +package xmpp + +import ( + "encoding/xml" +) + +// elementBuffer is a struct to store multiple values of type xml.Token until +// they form a complete XML element with opening and closing tag which is +// suitable to be passed to an appropriate handler function. +type elementBuffer struct { + tokens []xml.Token + end xml.EndElement + level int +} + +// newElementBuffer returns a new initialized elementBuffer struct. +func newElementBuffer() elementBuffer { + buf := elementBuffer{} + buf.reset() + return buf +} + +// FIXME this function needs essential error handling for corner cases! +// +// add is able to add a new xml.Token to the buffer. There are some rules +// checked to ensure a correct and consistent elementBuffer which are checked. +// If one of these checks fail the token is not added and a corresponding error +// is returned. +func (e *elementBuffer) add(t xml.Token) error { + switch t.(type) { + case xml.StartElement: + e.level += 1 + case xml.EndElement: + e.level -= 1 + } + e.tokens = append(e.tokens, t) + return nil +} + +// isComplete returns true if the buffer contains a slice of XML tokens which +// form a complete XML element starting with an xml.StartElement and closing +// with the corresponding xml.EndElement. +func (e *elementBuffer) isComplete() bool { + return (len(e.tokens) > 0 && e.level == 0) +} + +// reset returns the content of the buffer as a slice of XML tokens and resets +// the buffer to the initial state. This function can be used to initialize the +// elementBuffer struct. In that case the return value can be ignored. +func (e *elementBuffer) reset() (buf []xml.Token) { + retval := e.tokens + + e.tokens = make([]xml.Token, 0) + e.end = xml.EndElement{} + e.level = 0 + + return retval +} -- cgit v1.2.3-70-g09d2 From 3df1a88c726c08704e8b71c467bd8e11c9a52db6 Mon Sep 17 00:00:00 2001 From: xengineering Date: Thu, 29 Jun 2023 21:36:26 +0200 Subject: Add FIXME to element buffer specification The behaviour is ok for now but should be improved in the future to make it more robust. --- xmpp/element_buffer.go | 3 +++ 1 file changed, 3 insertions(+) (limited to 'xmpp/element_buffer.go') diff --git a/xmpp/element_buffer.go b/xmpp/element_buffer.go index c55279b..7792db2 100644 --- a/xmpp/element_buffer.go +++ b/xmpp/element_buffer.go @@ -37,6 +37,9 @@ func (e *elementBuffer) add(t xml.Token) error { return nil } +// FIXME isComplete would be true if a stream with only one XML comment is +// passed to the buffer. This might be unexpected behaviour. +// // isComplete returns true if the buffer contains a slice of XML tokens which // form a complete XML element starting with an xml.StartElement and closing // with the corresponding xml.EndElement. -- cgit v1.2.3-70-g09d2