webservices: Keep track of the parent-child relationship between nodes.
Signed-off-by: Hans Leidekker <hans@codeweavers.com> Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
parent
49638f1a7f
commit
ff990d3426
|
@ -254,6 +254,8 @@ struct node
|
||||||
{
|
{
|
||||||
WS_XML_ELEMENT_NODE hdr;
|
WS_XML_ELEMENT_NODE hdr;
|
||||||
struct list entry;
|
struct list entry;
|
||||||
|
struct node *parent;
|
||||||
|
struct list children;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct node *alloc_node( WS_XML_NODE_TYPE type )
|
static struct node *alloc_node( WS_XML_NODE_TYPE type )
|
||||||
|
@ -263,6 +265,7 @@ static struct node *alloc_node( WS_XML_NODE_TYPE type )
|
||||||
if (!(ret = heap_alloc_zero( sizeof(*ret) ))) return NULL;
|
if (!(ret = heap_alloc_zero( sizeof(*ret) ))) return NULL;
|
||||||
ret->hdr.node.nodeType = type;
|
ret->hdr.node.nodeType = type;
|
||||||
list_init( &ret->entry );
|
list_init( &ret->entry );
|
||||||
|
list_init( &ret->children );
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -316,16 +319,18 @@ static void free_node( struct node *node )
|
||||||
heap_free( node );
|
heap_free( node );
|
||||||
}
|
}
|
||||||
|
|
||||||
static void destroy_nodes( struct list *list )
|
static void destroy_nodes( struct node *node )
|
||||||
{
|
{
|
||||||
struct list *ptr;
|
struct list *ptr;
|
||||||
|
|
||||||
while ((ptr = list_head( list )))
|
if (!node) return;
|
||||||
|
while ((ptr = list_head( &node->children )))
|
||||||
{
|
{
|
||||||
struct node *node = LIST_ENTRY( ptr, struct node, entry );
|
struct node *child = LIST_ENTRY( ptr, struct node, entry );
|
||||||
list_remove( &node->entry );
|
list_remove( &child->entry );
|
||||||
free_node( node );
|
destroy_nodes( child );
|
||||||
}
|
}
|
||||||
|
free_node( node );
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct
|
static const struct
|
||||||
|
@ -369,7 +374,7 @@ struct reader
|
||||||
ULONG read_pos;
|
ULONG read_pos;
|
||||||
const char *read_bufptr;
|
const char *read_bufptr;
|
||||||
enum reader_state state;
|
enum reader_state state;
|
||||||
struct list nodes;
|
struct node *root;
|
||||||
struct node *current;
|
struct node *current;
|
||||||
WS_XML_READER_INPUT_TYPE input_type;
|
WS_XML_READER_INPUT_TYPE input_type;
|
||||||
const char *input_data;
|
const char *input_data;
|
||||||
|
@ -417,18 +422,45 @@ static HRESULT get_reader_prop( struct reader *reader, WS_XML_READER_PROPERTY_ID
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void read_insert_eof( struct reader *reader, struct node *eof )
|
||||||
|
{
|
||||||
|
if (!reader->root) reader->root = eof;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
eof->parent = reader->root;
|
||||||
|
list_add_tail( &reader->root->children, &eof->entry );
|
||||||
|
}
|
||||||
|
reader->current = eof;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void read_insert_bof( struct reader *reader, struct node *bof )
|
||||||
|
{
|
||||||
|
reader->root->parent = bof;
|
||||||
|
list_add_tail( &bof->children, &reader->root->entry );
|
||||||
|
reader->current = reader->root = bof;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void read_insert_node( struct reader *reader, struct node *parent, struct node *node )
|
||||||
|
{
|
||||||
|
node->parent = parent;
|
||||||
|
if (node->parent == reader->root)
|
||||||
|
{
|
||||||
|
struct list *eof = list_tail( &reader->root->children );
|
||||||
|
list_add_before( eof, &node->entry );
|
||||||
|
}
|
||||||
|
else list_add_tail( &parent->children, &node->entry );
|
||||||
|
reader->current = node;
|
||||||
|
}
|
||||||
|
|
||||||
static HRESULT read_init_state( struct reader *reader )
|
static HRESULT read_init_state( struct reader *reader )
|
||||||
{
|
{
|
||||||
struct node *node;
|
struct node *node;
|
||||||
|
|
||||||
list_init( &reader->nodes );
|
destroy_nodes( reader->root );
|
||||||
|
reader->root = NULL;
|
||||||
if (!(node = alloc_node( WS_XML_NODE_TYPE_EOF ))) return E_OUTOFMEMORY;
|
if (!(node = alloc_node( WS_XML_NODE_TYPE_EOF ))) return E_OUTOFMEMORY;
|
||||||
list_add_tail( &reader->nodes, &node->entry );
|
read_insert_eof( reader, node );
|
||||||
reader->current = node;
|
|
||||||
reader->state = READER_STATE_INITIAL;
|
reader->state = READER_STATE_INITIAL;
|
||||||
reader->read_size = 0;
|
|
||||||
reader->read_pos = 0;
|
|
||||||
reader->read_bufptr = NULL;
|
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -486,7 +518,7 @@ void WINAPI WsFreeReader( WS_XML_READER *handle )
|
||||||
TRACE( "%p\n", handle );
|
TRACE( "%p\n", handle );
|
||||||
|
|
||||||
if (!reader) return;
|
if (!reader) return;
|
||||||
destroy_nodes( &reader->nodes );
|
destroy_nodes( reader->root );
|
||||||
heap_free( reader );
|
heap_free( reader );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -848,7 +880,8 @@ static HRESULT read_element( struct reader *reader )
|
||||||
|
|
||||||
if (read_end_of_data( reader ))
|
if (read_end_of_data( reader ))
|
||||||
{
|
{
|
||||||
reader->current = LIST_ENTRY( list_tail( &reader->nodes ), struct node, entry );
|
struct list *eof = list_tail( &reader->root->children );
|
||||||
|
reader->current = LIST_ENTRY( eof, struct node, entry );
|
||||||
reader->state = READER_STATE_EOF;
|
reader->state = READER_STATE_EOF;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
@ -893,11 +926,11 @@ static HRESULT read_element( struct reader *reader )
|
||||||
hr = WS_E_INVALID_FORMAT;
|
hr = WS_E_INVALID_FORMAT;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
read_skip( reader, 1 );
|
|
||||||
|
|
||||||
list_add_after( &reader->current->entry, &node->entry );
|
read_insert_node( reader, reader->current, node );
|
||||||
reader->current = node;
|
read_skip( reader, 1 );
|
||||||
reader->state = READER_STATE_STARTELEMENT;
|
reader->state = READER_STATE_STARTELEMENT;
|
||||||
|
|
||||||
return S_OK;
|
return S_OK;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
|
@ -932,8 +965,7 @@ static HRESULT read_text( struct reader *reader )
|
||||||
}
|
}
|
||||||
text->text = (WS_XML_TEXT *)utf8;
|
text->text = (WS_XML_TEXT *)utf8;
|
||||||
|
|
||||||
list_add_after( &reader->current->entry, &node->entry );
|
read_insert_node( reader, reader->current, node );
|
||||||
reader->current = node;
|
|
||||||
reader->state = READER_STATE_TEXT;
|
reader->state = READER_STATE_TEXT;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
@ -951,10 +983,9 @@ static HRESULT read_to_startelement( struct reader *reader, BOOL *found )
|
||||||
switch (reader->state)
|
switch (reader->state)
|
||||||
{
|
{
|
||||||
case READER_STATE_INITIAL:
|
case READER_STATE_INITIAL:
|
||||||
{
|
|
||||||
if ((hr = read_xmldecl( reader )) != S_OK) return hr;
|
if ((hr = read_xmldecl( reader )) != S_OK) return hr;
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
case READER_STATE_STARTELEMENT:
|
case READER_STATE_STARTELEMENT:
|
||||||
if (found) *found = TRUE;
|
if (found) *found = TRUE;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
|
@ -975,16 +1006,48 @@ static HRESULT read_to_startelement( struct reader *reader, BOOL *found )
|
||||||
return hr;
|
return hr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static BOOL cmp_localname( const char *name1, ULONG len1, const char *name2, ULONG len2 )
|
||||||
|
{
|
||||||
|
ULONG i;
|
||||||
|
if (len1 != len2) return FALSE;
|
||||||
|
for (i = 0; i < len1; i++) { if (toupper( name1[i] ) != toupper( name2[i] )) return FALSE; }
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct node *find_parent_element( struct node *node, const char *localname, ULONG len )
|
||||||
|
{
|
||||||
|
struct node *parent;
|
||||||
|
WS_XML_STRING *name;
|
||||||
|
|
||||||
|
for (parent = node; parent; parent = parent->parent)
|
||||||
|
{
|
||||||
|
if (parent->hdr.node.nodeType != WS_XML_NODE_TYPE_ELEMENT) continue;
|
||||||
|
name = ((WS_XML_ELEMENT_NODE *)parent)->localName;
|
||||||
|
if (!cmp_localname( (const char *)name->bytes, name->length, localname, len )) continue;
|
||||||
|
return parent;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
static HRESULT read_endelement( struct reader *reader )
|
static HRESULT read_endelement( struct reader *reader )
|
||||||
{
|
{
|
||||||
struct node *node;
|
struct node *node, *parent;
|
||||||
unsigned int ch, skip;
|
unsigned int len = 0, ch, skip;
|
||||||
|
const char *start;
|
||||||
|
|
||||||
if (reader->state != READER_STATE_TEXT) return WS_E_INVALID_FORMAT;
|
switch (reader->state)
|
||||||
|
{
|
||||||
|
case READER_STATE_TEXT:
|
||||||
|
case READER_STATE_STARTELEMENT:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return WS_E_INVALID_FORMAT;
|
||||||
|
}
|
||||||
|
|
||||||
if (read_cmp( reader, "</", 2 )) return WS_E_INVALID_FORMAT;
|
if (read_cmp( reader, "</", 2 )) return WS_E_INVALID_FORMAT;
|
||||||
read_skip( reader, 2 );
|
read_skip( reader, 2 );
|
||||||
|
|
||||||
|
start = read_current_ptr( reader );
|
||||||
for (;;)
|
for (;;)
|
||||||
{
|
{
|
||||||
if (!(ch = read_utf8_char( reader, &skip ))) return WS_E_INVALID_FORMAT;
|
if (!(ch = read_utf8_char( reader, &skip ))) return WS_E_INVALID_FORMAT;
|
||||||
|
@ -995,11 +1058,13 @@ static HRESULT read_endelement( struct reader *reader )
|
||||||
}
|
}
|
||||||
if (!read_isnamechar( ch )) return WS_E_INVALID_FORMAT;
|
if (!read_isnamechar( ch )) return WS_E_INVALID_FORMAT;
|
||||||
read_skip( reader, skip );
|
read_skip( reader, skip );
|
||||||
|
len += skip;
|
||||||
}
|
}
|
||||||
|
if (!(parent = find_parent_element( reader->current, start, len )))
|
||||||
|
return WS_E_INVALID_FORMAT;
|
||||||
|
|
||||||
if (!(node = alloc_node( WS_XML_NODE_TYPE_END_ELEMENT ))) return E_OUTOFMEMORY;
|
if (!(node = alloc_node( WS_XML_NODE_TYPE_END_ELEMENT ))) return E_OUTOFMEMORY;
|
||||||
list_add_after( &reader->current->entry, &node->entry );
|
read_insert_node( reader, parent, node );
|
||||||
reader->current = node;
|
|
||||||
reader->state = READER_STATE_ENDELEMENT;
|
reader->state = READER_STATE_ENDELEMENT;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
@ -1037,8 +1102,7 @@ static HRESULT read_comment( struct reader *reader )
|
||||||
memcpy( comment->value.bytes, start, len );
|
memcpy( comment->value.bytes, start, len );
|
||||||
comment->value.length = len;
|
comment->value.length = len;
|
||||||
|
|
||||||
list_add_after( &reader->current->entry, &node->entry );
|
read_insert_node( reader, reader->current, node );
|
||||||
reader->current = node;
|
|
||||||
reader->state = READER_STATE_COMMENT;
|
reader->state = READER_STATE_COMMENT;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
@ -1051,7 +1115,8 @@ static HRESULT read_node( struct reader *reader )
|
||||||
{
|
{
|
||||||
if (read_end_of_data( reader ))
|
if (read_end_of_data( reader ))
|
||||||
{
|
{
|
||||||
reader->current = LIST_ENTRY( list_tail( &reader->nodes ), struct node, entry );
|
struct list *eof = list_tail( &reader->root->children );
|
||||||
|
reader->current = LIST_ENTRY( eof, struct node, entry );
|
||||||
reader->state = READER_STATE_EOF;
|
reader->state = READER_STATE_EOF;
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
@ -1447,6 +1512,17 @@ static WS_CHARSET detect_charset( const unsigned char *data, ULONG size, ULONG *
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void set_input_buffer( struct reader *reader, const char *data, ULONG size )
|
||||||
|
{
|
||||||
|
reader->input_type = WS_XML_READER_INPUT_TYPE_BUFFER;
|
||||||
|
reader->input_data = data;
|
||||||
|
reader->input_size = size;
|
||||||
|
|
||||||
|
reader->read_size = reader->input_size;
|
||||||
|
reader->read_pos = 0;
|
||||||
|
reader->read_bufptr = reader->input_data;
|
||||||
|
}
|
||||||
|
|
||||||
/**************************************************************************
|
/**************************************************************************
|
||||||
* WsSetInput [webservices.@]
|
* WsSetInput [webservices.@]
|
||||||
*/
|
*/
|
||||||
|
@ -1470,13 +1546,8 @@ HRESULT WINAPI WsSetInput( WS_XML_READER *handle, const WS_XML_READER_ENCODING *
|
||||||
if (hr != S_OK) return hr;
|
if (hr != S_OK) return hr;
|
||||||
}
|
}
|
||||||
|
|
||||||
destroy_nodes( &reader->nodes );
|
|
||||||
if ((hr = read_init_state( reader )) != S_OK) return hr;
|
if ((hr = read_init_state( reader )) != S_OK) return hr;
|
||||||
|
|
||||||
if (!(node = alloc_node( WS_XML_NODE_TYPE_BOF ))) return E_OUTOFMEMORY;
|
|
||||||
list_add_head( &reader->nodes, &node->entry );
|
|
||||||
reader->current = node;
|
|
||||||
|
|
||||||
switch (encoding->encodingType)
|
switch (encoding->encodingType)
|
||||||
{
|
{
|
||||||
case WS_XML_READER_ENCODING_TYPE_TEXT:
|
case WS_XML_READER_ENCODING_TYPE_TEXT:
|
||||||
|
@ -1507,11 +1578,7 @@ HRESULT WINAPI WsSetInput( WS_XML_READER *handle, const WS_XML_READER_ENCODING *
|
||||||
case WS_XML_READER_INPUT_TYPE_BUFFER:
|
case WS_XML_READER_INPUT_TYPE_BUFFER:
|
||||||
{
|
{
|
||||||
WS_XML_READER_BUFFER_INPUT *buf = (WS_XML_READER_BUFFER_INPUT *)input;
|
WS_XML_READER_BUFFER_INPUT *buf = (WS_XML_READER_BUFFER_INPUT *)input;
|
||||||
|
set_input_buffer( reader, (const char *)buf->encodedData + offset, buf->encodedDataSize - offset );
|
||||||
reader->input_type = WS_XML_READER_INPUT_TYPE_BUFFER;
|
|
||||||
reader->input_data = (const char *)buf->encodedData + offset;
|
|
||||||
reader->input_size = buf->encodedDataSize - offset;
|
|
||||||
reader->read_bufptr = reader->input_data;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
@ -1519,6 +1586,8 @@ HRESULT WINAPI WsSetInput( WS_XML_READER *handle, const WS_XML_READER_ENCODING *
|
||||||
return E_NOTIMPL;
|
return E_NOTIMPL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!(node = alloc_node( WS_XML_NODE_TYPE_BOF ))) return E_OUTOFMEMORY;
|
||||||
|
read_insert_bof( reader, node );
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1547,21 +1616,15 @@ HRESULT WINAPI WsSetInputToBuffer( WS_XML_READER *handle, WS_XML_BUFFER *buffer,
|
||||||
if (hr != S_OK) return hr;
|
if (hr != S_OK) return hr;
|
||||||
}
|
}
|
||||||
|
|
||||||
destroy_nodes( &reader->nodes );
|
|
||||||
if ((hr = read_init_state( reader )) != S_OK) return hr;
|
if ((hr = read_init_state( reader )) != S_OK) return hr;
|
||||||
|
|
||||||
if (!(node = alloc_node( WS_XML_NODE_TYPE_BOF ))) return E_OUTOFMEMORY;
|
|
||||||
list_add_head( &reader->nodes, &node->entry );
|
|
||||||
reader->current = node;
|
|
||||||
|
|
||||||
charset = detect_charset( xmlbuf->ptr, xmlbuf->size, &offset );
|
charset = detect_charset( xmlbuf->ptr, xmlbuf->size, &offset );
|
||||||
hr = set_reader_prop( reader, WS_XML_READER_PROPERTY_CHARSET, &charset, sizeof(charset) );
|
hr = set_reader_prop( reader, WS_XML_READER_PROPERTY_CHARSET, &charset, sizeof(charset) );
|
||||||
if (hr != S_OK) return hr;
|
if (hr != S_OK) return hr;
|
||||||
|
|
||||||
reader->input_type = WS_XML_READER_INPUT_TYPE_BUFFER;
|
set_input_buffer( reader, (const char *)xmlbuf->ptr + offset, xmlbuf->size - offset );
|
||||||
reader->input_data = (const char *)xmlbuf->ptr + offset;
|
if (!(node = alloc_node( WS_XML_NODE_TYPE_BOF ))) return E_OUTOFMEMORY;
|
||||||
reader->input_size = xmlbuf->size - offset;
|
read_insert_bof( reader, node );
|
||||||
reader->read_bufptr = reader->input_data;
|
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue