Commit d0413fd4 authored by Nikolay Sivov's avatar Nikolay Sivov Committed by Alexandre Julliard

msxml3: Fix empty elements output.

parent a43aa90d
...@@ -45,6 +45,7 @@ static const WCHAR utf16W[] = {'U','T','F','-','1','6',0}; ...@@ -45,6 +45,7 @@ static const WCHAR utf16W[] = {'U','T','F','-','1','6',0};
static const WCHAR utf8W[] = {'U','T','F','-','8',0}; static const WCHAR utf8W[] = {'U','T','F','-','8',0};
static const char crlfA[] = "\r\n"; static const char crlfA[] = "\r\n";
static const WCHAR emptyW[] = {0};
typedef enum typedef enum
{ {
...@@ -69,6 +70,10 @@ typedef struct _mxwriter ...@@ -69,6 +70,10 @@ typedef struct _mxwriter
xmlCharEncoding encoding; xmlCharEncoding encoding;
BSTR version; BSTR version;
/* contains a pending (or not closed yet) element name or NULL if
we don't have to close */
BSTR element;
IStream *dest; IStream *dest;
ULONG dest_written; ULONG dest_written;
...@@ -142,8 +147,24 @@ static HRESULT write_data_to_stream(mxwriter *This) ...@@ -142,8 +147,24 @@ static HRESULT write_data_to_stream(mxwriter *This)
return hres; return hres;
} }
/* Newly added element start tag left unclosed cause for empty elements
we have to close it differently. */
static void close_element_starttag(const mxwriter *This)
{
if (!This->element) return;
xmlOutputBufferWriteString(This->buffer, ">");
}
static void set_element_name(mxwriter *This, const WCHAR *name)
{
SysFreeString(This->element);
This->element = SysAllocString(name);
}
static inline HRESULT flush_output_buffer(mxwriter *This) static inline HRESULT flush_output_buffer(mxwriter *This)
{ {
close_element_starttag(This);
set_element_name(This, NULL);
xmlOutputBufferFlush(This->buffer); xmlOutputBufferFlush(This->buffer);
return write_data_to_stream(This); return write_data_to_stream(This);
} }
...@@ -221,6 +242,7 @@ static ULONG WINAPI mxwriter_Release(IMXWriter *iface) ...@@ -221,6 +242,7 @@ static ULONG WINAPI mxwriter_Release(IMXWriter *iface)
SysFreeString(This->version); SysFreeString(This->version);
xmlOutputBufferClose(This->buffer); xmlOutputBufferClose(This->buffer);
SysFreeString(This->element);
heap_free(This); heap_free(This);
} }
...@@ -740,6 +762,9 @@ static HRESULT WINAPI mxwriter_saxcontent_startElement( ...@@ -740,6 +762,9 @@ static HRESULT WINAPI mxwriter_saxcontent_startElement(
if ((!namespaceUri || !local_name || !QName) && This->class_version != MSXML6) if ((!namespaceUri || !local_name || !QName) && This->class_version != MSXML6)
return E_INVALIDARG; return E_INVALIDARG;
close_element_starttag(This);
set_element_name(This, QName ? QName : emptyW);
xmlOutputBufferWriteString(This->buffer, "<"); xmlOutputBufferWriteString(This->buffer, "<");
s = xmlchar_from_wchar(QName); s = xmlchar_from_wchar(QName);
xmlOutputBufferWriteString(This->buffer, (char*)s); xmlOutputBufferWriteString(This->buffer, (char*)s);
...@@ -782,8 +807,6 @@ static HRESULT WINAPI mxwriter_saxcontent_startElement( ...@@ -782,8 +807,6 @@ static HRESULT WINAPI mxwriter_saxcontent_startElement(
} }
} }
xmlOutputBufferWriteString(This->buffer, ">");
return S_OK; return S_OK;
} }
...@@ -805,11 +828,21 @@ static HRESULT WINAPI mxwriter_saxcontent_endElement( ...@@ -805,11 +828,21 @@ static HRESULT WINAPI mxwriter_saxcontent_endElement(
if ((!namespaceUri || !local_name || !QName) && This->class_version != MSXML6) if ((!namespaceUri || !local_name || !QName) && This->class_version != MSXML6)
return E_INVALIDARG; return E_INVALIDARG;
xmlOutputBufferWriteString(This->buffer, "</");
s = xmlchar_from_wchar(QName); s = xmlchar_from_wchar(QName);
xmlOutputBufferWriteString(This->buffer, (char*)s);
if (This->element && QName && !strcmpW(This->element, QName))
{
xmlOutputBufferWriteString(This->buffer, "/>");
}
else
{
xmlOutputBufferWriteString(This->buffer, "</");
xmlOutputBufferWriteString(This->buffer, (char*)s);
xmlOutputBufferWriteString(This->buffer, ">");
}
heap_free(s); heap_free(s);
xmlOutputBufferWriteString(This->buffer, ">"); set_element_name(This, NULL);
return S_OK; return S_OK;
} }
...@@ -825,6 +858,9 @@ static HRESULT WINAPI mxwriter_saxcontent_characters( ...@@ -825,6 +858,9 @@ static HRESULT WINAPI mxwriter_saxcontent_characters(
if (!chars) return E_INVALIDARG; if (!chars) return E_INVALIDARG;
close_element_starttag(This);
set_element_name(This, NULL);
if (nchars) if (nchars)
{ {
xmlChar *s = xmlchar_from_wcharn(chars, nchars); xmlChar *s = xmlchar_from_wcharn(chars, nchars);
...@@ -912,6 +948,8 @@ HRESULT MXWriter_create(MSXML_VERSION version, IUnknown *pUnkOuter, void **ppObj ...@@ -912,6 +948,8 @@ HRESULT MXWriter_create(MSXML_VERSION version, IUnknown *pUnkOuter, void **ppObj
This->encoding = xmlParseCharEncoding("UTF-16"); This->encoding = xmlParseCharEncoding("UTF-16");
This->version = SysAllocString(version10W); This->version = SysAllocString(version10W);
This->element = NULL;
This->dest = NULL; This->dest = NULL;
This->dest_written = 0; This->dest_written = 0;
......
...@@ -51,7 +51,7 @@ static BSTR alloc_str_from_narrow(const char *str) ...@@ -51,7 +51,7 @@ static BSTR alloc_str_from_narrow(const char *str)
return ret; return ret;
} }
static BSTR alloced_bstrs[256]; static BSTR alloced_bstrs[512];
static int alloced_bstrs_count; static int alloced_bstrs_count;
static BSTR _bstr_(const char *str) static BSTR _bstr_(const char *str)
...@@ -1467,7 +1467,8 @@ static void test_mxwriter_startenddocument(void) ...@@ -1467,7 +1467,8 @@ static void test_mxwriter_startenddocument(void)
enum startendtype enum startendtype
{ {
StartElement, StartElement,
EndElement EndElement,
StartEndElement
}; };
struct writer_startendelement_t { struct writer_startendelement_t {
...@@ -1567,6 +1568,17 @@ static const struct writer_startendelement_t writer_startendelement[] = { ...@@ -1567,6 +1568,17 @@ static const struct writer_startendelement_t writer_startendelement[] = {
{ &CLSID_MXXMLWriter30, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes }, { &CLSID_MXXMLWriter30, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
{ &CLSID_MXXMLWriter40, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes }, { &CLSID_MXXMLWriter40, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
{ &CLSID_MXXMLWriter60, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes }, { &CLSID_MXXMLWriter60, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
/* empty elements */
{ &CLSID_MXXMLWriter, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
{ &CLSID_MXXMLWriter30, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
/* 70 */
{ &CLSID_MXXMLWriter40, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
{ &CLSID_MXXMLWriter60, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
{ &CLSID_MXXMLWriter, StartEndElement, "", "", "", "</>", S_OK },
{ &CLSID_MXXMLWriter30, StartEndElement, "", "", "", "</>", S_OK },
{ &CLSID_MXXMLWriter40, StartEndElement, "", "", "", "</>", S_OK },
/* 75 */
{ &CLSID_MXXMLWriter60, StartEndElement, "", "", "", "</>", S_OK },
{ NULL } { NULL }
}; };
...@@ -1619,12 +1631,26 @@ static void test_mxwriter_startendelement_batch(const struct writer_startendelem ...@@ -1619,12 +1631,26 @@ static void test_mxwriter_startendelement_batch(const struct writer_startendelem
EXPECT_HR(hr, S_OK); EXPECT_HR(hr, S_OK);
if (table->type == StartElement) if (table->type == StartElement)
{
hr = ISAXContentHandler_startElement(content, _bstr_(table->uri), lstrlen(table->uri), hr = ISAXContentHandler_startElement(content, _bstr_(table->uri), lstrlen(table->uri),
_bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname), table->attr); _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname), table->attr);
ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
}
else if (table->type == EndElement)
{
hr = ISAXContentHandler_endElement(content, _bstr_(table->uri), lstrlen(table->uri),
_bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname));
ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
}
else else
{
hr = ISAXContentHandler_startElement(content, _bstr_(table->uri), lstrlen(table->uri),
_bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname), table->attr);
ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
hr = ISAXContentHandler_endElement(content, _bstr_(table->uri), lstrlen(table->uri), hr = ISAXContentHandler_endElement(content, _bstr_(table->uri), lstrlen(table->uri),
_bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname)); _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname));
ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr); ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
}
/* test output */ /* test output */
if (hr == S_OK) if (hr == S_OK)
...@@ -1652,7 +1678,6 @@ static void test_mxwriter_startendelement_batch(const struct writer_startendelem ...@@ -1652,7 +1678,6 @@ static void test_mxwriter_startendelement_batch(const struct writer_startendelem
static void test_mxwriter_startendelement(void) static void test_mxwriter_startendelement(void)
{ {
static const char winehqA[] = "http://winehq.org";
ISAXContentHandler *content; ISAXContentHandler *content;
IMXWriter *writer; IMXWriter *writer;
VARIANT dest; VARIANT dest;
...@@ -1714,31 +1739,6 @@ static void test_mxwriter_startendelement(void) ...@@ -1714,31 +1739,6 @@ static void test_mxwriter_startendelement(void)
ok(!lstrcmpW(_bstr_("<><b></b>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest))); ok(!lstrcmpW(_bstr_("<><b></b>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
VariantClear(&dest); VariantClear(&dest);
/* some with namespace URI */
hr = ISAXContentHandler_startElement(content, _bstr_(winehqA), sizeof(winehqA), _bstr_(""), 0, _bstr_("nspace:c"), 8, NULL);
ok(hr == S_OK, "got %08x\n", hr);
hr = ISAXContentHandler_endElement(content, _bstr_(winehqA), sizeof(winehqA), _bstr_(""), 0, _bstr_("nspace:c"), 8);
ok(hr == S_OK, "got %08x\n", hr);
V_VT(&dest) = VT_EMPTY;
hr = IMXWriter_get_output(writer, &dest);
ok(hr == S_OK, "got %08x\n", hr);
ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
todo_wine ok(!lstrcmpW(_bstr_("<><b></b><nspace:c/>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
VariantClear(&dest);
/* try to end element that wasn't open */
hr = ISAXContentHandler_endElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("a"), 1);
ok(hr == S_OK, "got %08x\n", hr);
V_VT(&dest) = VT_EMPTY;
hr = IMXWriter_get_output(writer, &dest);
ok(hr == S_OK, "got %08x\n", hr);
ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
todo_wine ok(!lstrcmpW(_bstr_("<><b></b><nspace:c/></a>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
VariantClear(&dest);
hr = ISAXContentHandler_endDocument(content); hr = ISAXContentHandler_endDocument(content);
ok(hr == S_OK, "got %08x\n", hr); ok(hr == S_OK, "got %08x\n", hr);
...@@ -1758,35 +1758,68 @@ static void test_mxwriter_characters(void) ...@@ -1758,35 +1758,68 @@ static void test_mxwriter_characters(void)
hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER, hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
&IID_IMXWriter, (void**)&writer); &IID_IMXWriter, (void**)&writer);
ok(hr == S_OK, "Expected S_OK, got %08x\n", hr); EXPECT_HR(hr, S_OK);
hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content); hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE); hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_startDocument(content); hr = ISAXContentHandler_startDocument(content);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_characters(content, NULL, 0); hr = ISAXContentHandler_characters(content, NULL, 0);
ok(hr == E_INVALIDARG, "got %08x\n", hr); EXPECT_HR(hr, E_INVALIDARG);
hr = ISAXContentHandler_characters(content, chardataW, 0); hr = ISAXContentHandler_characters(content, chardataW, 0);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_characters(content, chardataW, sizeof(chardataW)/sizeof(WCHAR) - 1); hr = ISAXContentHandler_characters(content, chardataW, sizeof(chardataW)/sizeof(WCHAR) - 1);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
V_VT(&dest) = VT_EMPTY; V_VT(&dest) = VT_EMPTY;
hr = IMXWriter_get_output(writer, &dest); hr = IMXWriter_get_output(writer, &dest);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest)); ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
ok(!lstrcmpW(_bstr_("TESTCHARDATA ."), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest))); ok(!lstrcmpW(_bstr_("TESTCHARDATA ."), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
VariantClear(&dest); VariantClear(&dest);
hr = ISAXContentHandler_endDocument(content); hr = ISAXContentHandler_endDocument(content);
ok(hr == S_OK, "got %08x\n", hr); EXPECT_HR(hr, S_OK);
ISAXContentHandler_Release(content);
IMXWriter_Release(writer);
/* try empty characters data to see if element is closed */
hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
&IID_IMXWriter, (void**)&writer);
EXPECT_HR(hr, S_OK);
hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
EXPECT_HR(hr, S_OK);
hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_startDocument(content);
EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_startElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("a"), 1, NULL);
EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_characters(content, chardataW, 0);
EXPECT_HR(hr, S_OK);
hr = ISAXContentHandler_endElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("a"), 1);
EXPECT_HR(hr, S_OK);
V_VT(&dest) = VT_EMPTY;
hr = IMXWriter_get_output(writer, &dest);
EXPECT_HR(hr, S_OK);
ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
ok(!lstrcmpW(_bstr_("<a></a>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
VariantClear(&dest);
ISAXContentHandler_Release(content); ISAXContentHandler_Release(content);
IMXWriter_Release(writer); IMXWriter_Release(writer);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment