2012-04-29 18:52:25 +00:00
|
|
|
//
|
|
|
|
// XMLWriter.cpp
|
|
|
|
//
|
|
|
|
// Library: XML
|
|
|
|
// Package: XML
|
|
|
|
// Module: XMLWriter
|
|
|
|
//
|
|
|
|
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
2014-05-04 21:02:42 +02:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
2012-04-29 18:52:25 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#include "Poco/XML/XMLWriter.h"
|
|
|
|
#include "Poco/XML/XMLString.h"
|
|
|
|
#include "Poco/XML/XMLException.h"
|
|
|
|
#include "Poco/SAX/AttributesImpl.h"
|
|
|
|
#include "Poco/UTF8Encoding.h"
|
|
|
|
#include "Poco/UTF16Encoding.h"
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
|
2024-01-30 09:35:35 +01:00
|
|
|
using namespace std::string_literals;
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
namespace Poco {
|
|
|
|
namespace XML {
|
|
|
|
|
|
|
|
|
|
|
|
const std::string XMLWriter::NEWLINE_DEFAULT;
|
|
|
|
const std::string XMLWriter::NEWLINE_CR = "\r";
|
|
|
|
const std::string XMLWriter::NEWLINE_CRLF = "\r\n";
|
|
|
|
const std::string XMLWriter::NEWLINE_LF = "\n";
|
|
|
|
const std::string XMLWriter::MARKUP_QUOTENC = """;
|
|
|
|
const std::string XMLWriter::MARKUP_AMPENC = "&";
|
|
|
|
const std::string XMLWriter::MARKUP_LTENC = "<";
|
|
|
|
const std::string XMLWriter::MARKUP_GTENC = ">";
|
|
|
|
const std::string XMLWriter::MARKUP_TABENC = "	";
|
|
|
|
const std::string XMLWriter::MARKUP_CRENC = "
";
|
|
|
|
const std::string XMLWriter::MARKUP_LFENC = "
";
|
|
|
|
const std::string XMLWriter::MARKUP_LT = "<";
|
|
|
|
const std::string XMLWriter::MARKUP_GT = ">";
|
|
|
|
const std::string XMLWriter::MARKUP_SLASHGT = "/>";
|
|
|
|
const std::string XMLWriter::MARKUP_LTSLASH = "</";
|
|
|
|
const std::string XMLWriter::MARKUP_COLON = ":";
|
|
|
|
const std::string XMLWriter::MARKUP_EQQUOT = "=\"";
|
|
|
|
const std::string XMLWriter::MARKUP_QUOT = "\"";
|
|
|
|
const std::string XMLWriter::MARKUP_SPACE = " ";
|
|
|
|
const std::string XMLWriter::MARKUP_TAB = "\t";
|
|
|
|
const std::string XMLWriter::MARKUP_BEGIN_CDATA = "<![CDATA[";
|
|
|
|
const std::string XMLWriter::MARKUP_END_CDATA = "]]>";
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(XML_UNICODE_WCHAR_T)
|
|
|
|
#define NATIVE_ENCODING Poco::UTF16Encoding
|
|
|
|
#else
|
|
|
|
#define NATIVE_ENCODING Poco::UTF8Encoding
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::XMLWriter(XMLByteOutputStream& str, int options):
|
|
|
|
_pTextConverter(0),
|
|
|
|
_pInEncoding(new NATIVE_ENCODING),
|
|
|
|
_pOutEncoding(new Poco::UTF8Encoding),
|
|
|
|
_options(options),
|
2024-01-30 09:35:35 +01:00
|
|
|
_encoding("UTF-8"s),
|
2012-04-29 18:52:25 +00:00
|
|
|
_depth(-1),
|
|
|
|
_elementCount(0),
|
|
|
|
_inFragment(false),
|
|
|
|
_inCDATA(false),
|
|
|
|
_inDTD(false),
|
|
|
|
_inInternalDTD(false),
|
|
|
|
_contentWritten(false),
|
|
|
|
_unclosedStartTag(false),
|
|
|
|
_prefix(0),
|
|
|
|
_nsContextPushed(false),
|
|
|
|
_indent(MARKUP_TAB)
|
|
|
|
{
|
|
|
|
_pTextConverter = new Poco::OutputStreamConverter(str, *_pInEncoding, *_pOutEncoding);
|
|
|
|
setNewLine((_options & CANONICAL_XML) ? NEWLINE_LF : NEWLINE_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::XMLWriter(XMLByteOutputStream& str, int options, const std::string& encodingName, Poco::TextEncoding& textEncoding):
|
|
|
|
_pTextConverter(0),
|
|
|
|
_pInEncoding(new NATIVE_ENCODING),
|
|
|
|
_pOutEncoding(0),
|
|
|
|
_options(options),
|
|
|
|
_encoding(encodingName),
|
|
|
|
_depth(-1),
|
|
|
|
_elementCount(0),
|
|
|
|
_inFragment(false),
|
|
|
|
_inCDATA(false),
|
|
|
|
_inDTD(false),
|
|
|
|
_inInternalDTD(false),
|
|
|
|
_contentWritten(false),
|
|
|
|
_unclosedStartTag(false),
|
|
|
|
_prefix(0),
|
|
|
|
_nsContextPushed(false),
|
|
|
|
_indent(MARKUP_TAB)
|
|
|
|
{
|
|
|
|
_pTextConverter = new Poco::OutputStreamConverter(str, *_pInEncoding, textEncoding);
|
|
|
|
setNewLine((_options & CANONICAL_XML) ? NEWLINE_LF : NEWLINE_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::XMLWriter(XMLByteOutputStream& str, int options, const std::string& encodingName, Poco::TextEncoding* pTextEncoding):
|
|
|
|
_pTextConverter(0),
|
|
|
|
_pInEncoding(new NATIVE_ENCODING),
|
|
|
|
_pOutEncoding(0),
|
|
|
|
_options(options),
|
|
|
|
_encoding(encodingName),
|
|
|
|
_depth(-1),
|
|
|
|
_elementCount(0),
|
|
|
|
_inFragment(false),
|
|
|
|
_inCDATA(false),
|
|
|
|
_inDTD(false),
|
|
|
|
_inInternalDTD(false),
|
|
|
|
_contentWritten(false),
|
|
|
|
_unclosedStartTag(false),
|
|
|
|
_prefix(0),
|
|
|
|
_nsContextPushed(false),
|
|
|
|
_indent(MARKUP_TAB)
|
|
|
|
{
|
|
|
|
if (pTextEncoding)
|
|
|
|
{
|
|
|
|
_pTextConverter = new Poco::OutputStreamConverter(str, *_pInEncoding, *pTextEncoding);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_encoding = "UTF-8";
|
|
|
|
_pOutEncoding = new Poco::UTF8Encoding;
|
|
|
|
_pTextConverter = new Poco::OutputStreamConverter(str, *_pInEncoding, *_pOutEncoding);
|
|
|
|
}
|
|
|
|
setNewLine((_options & CANONICAL_XML) ? NEWLINE_LF : NEWLINE_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::~XMLWriter()
|
|
|
|
{
|
|
|
|
delete _pTextConverter;
|
|
|
|
delete _pInEncoding;
|
|
|
|
delete _pOutEncoding;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::setDocumentLocator(const Locator* loc)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::setNewLine(const std::string& newLineCharacters)
|
|
|
|
{
|
|
|
|
if (newLineCharacters.empty())
|
|
|
|
{
|
|
|
|
#if defined(_WIN32)
|
|
|
|
_newLine = NEWLINE_CRLF;
|
|
|
|
#else
|
|
|
|
_newLine = NEWLINE_LF;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else _newLine = newLineCharacters;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const std::string& XMLWriter::getNewLine() const
|
|
|
|
{
|
|
|
|
return _newLine;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::setIndent(const std::string& indent)
|
|
|
|
{
|
|
|
|
_indent = indent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const std::string& XMLWriter::getIndent() const
|
|
|
|
{
|
|
|
|
return _indent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startDocument()
|
|
|
|
{
|
|
|
|
if (_depth != -1)
|
|
|
|
throw XMLException("Cannot start a document in another document");
|
|
|
|
|
|
|
|
_inFragment = false;
|
|
|
|
_depth = 0;
|
|
|
|
_elementCount = 0;
|
|
|
|
_inDTD = false;
|
|
|
|
_inInternalDTD = false;
|
|
|
|
_prefix = 0;
|
|
|
|
|
|
|
|
if (_options & WRITE_XML_DECLARATION)
|
|
|
|
writeXMLDeclaration();
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
_contentWritten = true;
|
|
|
|
_namespaces.reset();
|
|
|
|
_namespaces.pushContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endDocument()
|
|
|
|
{
|
|
|
|
if (_depth > 0)
|
|
|
|
throw XMLException("Not well-formed (at least one tag has no matching end tag)");
|
|
|
|
if (_elementCount == 0)
|
|
|
|
throw XMLException("No document element");
|
|
|
|
|
|
|
|
poco_assert_dbg (!_unclosedStartTag);
|
|
|
|
|
|
|
|
_elementCount = 0;
|
|
|
|
_depth = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startFragment()
|
|
|
|
{
|
|
|
|
if (_depth != -1)
|
|
|
|
throw XMLException("Cannot start a fragment in another fragment or document");
|
|
|
|
|
|
|
|
_inFragment = true;
|
|
|
|
_depth = 0;
|
|
|
|
_elementCount = 0;
|
|
|
|
_prefix = 0;
|
|
|
|
|
|
|
|
_contentWritten = true;
|
|
|
|
_namespaces.reset();
|
|
|
|
_namespaces.pushContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endFragment()
|
|
|
|
{
|
|
|
|
if (_depth > 1)
|
|
|
|
throw XMLException("Not well-formed (at least one tag has no matching end tag)");
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
_inFragment = false;
|
|
|
|
_elementCount = 0;
|
|
|
|
_depth = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname)
|
|
|
|
{
|
|
|
|
const AttributesImpl attributes;
|
|
|
|
startElement(namespaceURI, localName, qname, attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname, const Attributes& attributes)
|
|
|
|
{
|
2022-07-07 17:18:20 +08:00
|
|
|
if (_depth == 0 && !_inFragment && _elementCount > 1)
|
2012-04-29 18:52:25 +00:00
|
|
|
throw XMLException("Not well-formed. Second root element found", nameToString(localName, qname));
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
prettyPrint();
|
2016-02-28 11:20:02 +01:00
|
|
|
if (_options & CANONICAL_XML)
|
|
|
|
writeCanonicalStartElement(namespaceURI, localName, qname, attributes);
|
2022-07-07 17:18:20 +08:00
|
|
|
else
|
2016-02-28 11:20:02 +01:00
|
|
|
writeStartElement(namespaceURI, localName, qname, attributes);
|
2020-01-09 10:08:09 +01:00
|
|
|
_elementStack.emplace_back(qname, namespaceURI, localName);
|
2012-04-29 18:52:25 +00:00
|
|
|
_contentWritten = false;
|
|
|
|
++_depth;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname)
|
|
|
|
{
|
|
|
|
if (_depth < 1)
|
|
|
|
throw XMLException("No unclosed tag");
|
|
|
|
|
|
|
|
if (!_elementStack.back().equalsWeakly(qname, namespaceURI, localName))
|
|
|
|
throw XMLException("End tag does not match start tag", nameToString(localName, qname));
|
|
|
|
|
|
|
|
_elementStack.pop_back();
|
|
|
|
--_depth;
|
|
|
|
if (!_unclosedStartTag) prettyPrint();
|
|
|
|
writeEndElement(namespaceURI, localName, qname);
|
|
|
|
_contentWritten = false;
|
|
|
|
if (_depth == 0)
|
|
|
|
writeNewLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::emptyElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname)
|
|
|
|
{
|
|
|
|
const AttributesImpl attributes;
|
|
|
|
emptyElement(namespaceURI, localName, qname, attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::emptyElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname, const Attributes& attributes)
|
|
|
|
{
|
|
|
|
if (_depth == 0 && _elementCount > 1)
|
|
|
|
throw XMLException("Not well-formed. Second root element found.");
|
|
|
|
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
prettyPrint();
|
2016-02-28 11:20:02 +01:00
|
|
|
if (_options & CANONICAL_XML)
|
|
|
|
writeCanonicalStartElement(namespaceURI, localName, qname, attributes);
|
|
|
|
else
|
|
|
|
writeStartElement(namespaceURI, localName, qname, attributes);
|
2012-04-29 18:52:25 +00:00
|
|
|
_contentWritten = false;
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("/"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
closeStartTag();
|
2012-11-10 11:49:39 +01:00
|
|
|
_namespaces.popContext();
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::characters(const XMLChar ch[], int start, int length)
|
|
|
|
{
|
|
|
|
if (length == 0) return;
|
|
|
|
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
_contentWritten = _contentWritten || length > 0;
|
|
|
|
if (_inCDATA)
|
|
|
|
{
|
|
|
|
while (length-- > 0) writeXML(ch[start++]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (length-- > 0)
|
|
|
|
{
|
|
|
|
XMLChar c = ch[start++];
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case '"': writeMarkup(MARKUP_QUOTENC); break;
|
|
|
|
case '&': writeMarkup(MARKUP_AMPENC); break;
|
|
|
|
case '<': writeMarkup(MARKUP_LTENC); break;
|
|
|
|
case '>': writeMarkup(MARKUP_GTENC); break;
|
|
|
|
default:
|
|
|
|
if (c >= 0 && c < 32)
|
|
|
|
{
|
|
|
|
if (c == '\t' || c == '\r' || c == '\n')
|
|
|
|
writeXML(c);
|
|
|
|
else
|
|
|
|
throw XMLException("Invalid character token.");
|
|
|
|
}
|
|
|
|
else writeXML(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::characters(const XMLString& str)
|
|
|
|
{
|
|
|
|
characters(str.data(), 0, (int) str.length());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::rawCharacters(const XMLString& str)
|
|
|
|
{
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
_contentWritten = _contentWritten || !str.empty();
|
|
|
|
writeXML(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::ignorableWhitespace(const XMLChar ch[], int start, int length)
|
|
|
|
{
|
|
|
|
characters(ch, start, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::processingInstruction(const XMLString& target, const XMLString& data)
|
|
|
|
{
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
prettyPrint();
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<?"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(target);
|
|
|
|
if (!data.empty())
|
|
|
|
{
|
|
|
|
writeMarkup(MARKUP_SPACE);
|
|
|
|
writeXML(data);
|
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("?>"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
if (_depth == 0)
|
|
|
|
writeNewLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
static const XMLString CDATA = toXMLString("CDATA"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::dataElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname,
|
|
|
|
const XMLString& data,
|
|
|
|
const XMLString& attr1, const XMLString& value1,
|
|
|
|
const XMLString& attr2, const XMLString& value2,
|
|
|
|
const XMLString& attr3, const XMLString& value3)
|
|
|
|
{
|
|
|
|
AttributesImpl attributes;
|
|
|
|
if (!attr1.empty()) attributes.addAttribute(XMLString(), XMLString(), attr1, CDATA, value1);
|
|
|
|
if (!attr2.empty()) attributes.addAttribute(XMLString(), XMLString(), attr2, CDATA, value2);
|
|
|
|
if (!attr3.empty()) attributes.addAttribute(XMLString(), XMLString(), attr3, CDATA, value3);
|
|
|
|
if (data.empty())
|
|
|
|
{
|
|
|
|
emptyElement(namespaceURI, localName, qname, attributes);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
startElement(namespaceURI, localName, qname, attributes);
|
|
|
|
characters(data);
|
|
|
|
endElement(namespaceURI, localName, qname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startPrefixMapping(const XMLString& prefix, const XMLString& namespaceURI)
|
|
|
|
{
|
|
|
|
if (prefix != NamespaceSupport::XML_NAMESPACE_PREFIX)
|
|
|
|
{
|
|
|
|
if (!_nsContextPushed)
|
|
|
|
{
|
|
|
|
_namespaces.pushContext();
|
|
|
|
_nsContextPushed = true;
|
|
|
|
}
|
|
|
|
_namespaces.declarePrefix(prefix, namespaceURI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endPrefixMapping(const XMLString& prefix)
|
|
|
|
{
|
|
|
|
// Note: prefix removed by popContext() at element closing tag
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::skippedEntity(const XMLString& name)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startCDATA()
|
|
|
|
{
|
|
|
|
if (_inCDATA) throw XMLException("Cannot nest CDATA sections");
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
_inCDATA = true;
|
|
|
|
writeMarkup(MARKUP_BEGIN_CDATA);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endCDATA()
|
|
|
|
{
|
|
|
|
poco_assert (_inCDATA);
|
|
|
|
_inCDATA = false;
|
|
|
|
writeMarkup(MARKUP_END_CDATA);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::comment(const XMLChar ch[], int start, int length)
|
|
|
|
{
|
|
|
|
if (_unclosedStartTag) closeStartTag();
|
|
|
|
prettyPrint();
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<!--"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
while (length-- > 0) writeXML(ch[start++]);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("-->"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
_contentWritten = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startDTD(const XMLString& name, const XMLString& publicId, const XMLString& systemId)
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<!DOCTYPE "s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(name);
|
|
|
|
if (!publicId.empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" PUBLIC \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(publicId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
if (!systemId.empty())
|
|
|
|
{
|
2013-05-24 23:13:03 +02:00
|
|
|
if (publicId.empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" SYSTEM"s);
|
2013-05-24 23:13:03 +02:00
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(systemId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
_inDTD = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endDTD()
|
|
|
|
{
|
|
|
|
poco_assert (_inDTD);
|
|
|
|
if (_inInternalDTD)
|
|
|
|
{
|
|
|
|
writeNewLine();
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("]"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
_inInternalDTD = false;
|
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(">"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeNewLine();
|
|
|
|
_inDTD = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::startEntity(const XMLString& name)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::endEntity(const XMLString& name)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::notationDecl(const XMLString& name, const XMLString* publicId, const XMLString* systemId)
|
|
|
|
{
|
|
|
|
if (!_inDTD) throw XMLException("Notation declaration not within DTD");
|
|
|
|
if (!_inInternalDTD)
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" ["s);
|
2012-04-29 18:52:25 +00:00
|
|
|
_inInternalDTD = true;
|
|
|
|
}
|
|
|
|
if (_options & PRETTY_PRINT)
|
|
|
|
{
|
|
|
|
writeNewLine();
|
|
|
|
writeMarkup(_indent);
|
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<!NOTATION "s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(name);
|
|
|
|
if (systemId && !systemId->empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" SYSTEM \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(*systemId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
if (publicId && !publicId->empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" PUBLIC \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(*publicId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(">"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::unparsedEntityDecl(const XMLString& name, const XMLString* publicId, const XMLString& systemId, const XMLString& notationName)
|
|
|
|
{
|
|
|
|
if (!_inDTD) throw XMLException("Entity declaration not within DTD");
|
|
|
|
if (!_inInternalDTD)
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" ["s);
|
2012-04-29 18:52:25 +00:00
|
|
|
_inInternalDTD = true;
|
|
|
|
}
|
|
|
|
if (_options & PRETTY_PRINT)
|
|
|
|
{
|
|
|
|
writeNewLine();
|
|
|
|
writeMarkup(_indent);
|
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<!ENTITY "s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(name);
|
|
|
|
if (!systemId.empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" SYSTEM \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(systemId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
if (publicId && !publicId->empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" PUBLIC \""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(*publicId);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
if (!notationName.empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" NDATA "s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(notationName);
|
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(">"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::prettyPrint() const
|
|
|
|
{
|
|
|
|
if ((_options & PRETTY_PRINT) && !_contentWritten)
|
|
|
|
{
|
|
|
|
writeNewLine();
|
|
|
|
writeIndent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeStartElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname, const Attributes& attributes)
|
|
|
|
{
|
|
|
|
if (!_nsContextPushed)
|
|
|
|
_namespaces.pushContext();
|
|
|
|
_nsContextPushed = false;
|
|
|
|
++_elementCount;
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
declareAttributeNamespaces(attributes);
|
|
|
|
|
|
|
|
writeMarkup(MARKUP_LT);
|
|
|
|
if (!localName.empty() && (qname.empty() || localName == qname))
|
|
|
|
{
|
|
|
|
XMLString prefix;
|
|
|
|
if (!namespaceURI.empty() && !_namespaces.isMapped(namespaceURI))
|
|
|
|
{
|
2012-11-10 11:49:39 +01:00
|
|
|
prefix = uniquePrefix();
|
2012-04-29 18:52:25 +00:00
|
|
|
_namespaces.declarePrefix(prefix, namespaceURI);
|
|
|
|
}
|
|
|
|
else prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
writeName(prefix, localName);
|
|
|
|
}
|
|
|
|
else if (namespaceURI.empty() && localName.empty() && !qname.empty())
|
|
|
|
{
|
|
|
|
writeXML(qname);
|
|
|
|
}
|
|
|
|
else if (!localName.empty() && !qname.empty())
|
|
|
|
{
|
|
|
|
XMLString local;
|
|
|
|
XMLString prefix;
|
|
|
|
Name::split(qname, prefix, local);
|
|
|
|
if (prefix.empty()) prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
const XMLString& uri = _namespaces.getURI(prefix);
|
|
|
|
if ((uri.empty() || uri != namespaceURI) && !namespaceURI.empty())
|
|
|
|
{
|
|
|
|
_namespaces.declarePrefix(prefix, namespaceURI);
|
|
|
|
}
|
|
|
|
writeName(prefix, localName);
|
|
|
|
}
|
|
|
|
else throw XMLException("Tag mismatch", nameToString(localName, qname));
|
|
|
|
|
|
|
|
AttributeMap attributeMap;
|
|
|
|
addNamespaceAttributes(attributeMap);
|
|
|
|
addAttributes(attributeMap, attributes, namespaceURI);
|
|
|
|
writeAttributes(attributeMap);
|
|
|
|
_unclosedStartTag = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
void XMLWriter::writeCanonicalStartElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname, const Attributes& attributes)
|
|
|
|
{
|
|
|
|
if (!_nsContextPushed)
|
|
|
|
_namespaces.pushContext();
|
|
|
|
_nsContextPushed = false;
|
|
|
|
++_elementCount;
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
declareNamespaces(namespaceURI, localName, qname, attributes);
|
|
|
|
|
|
|
|
writeMarkup(MARKUP_LT);
|
|
|
|
if (!localName.empty())
|
|
|
|
{
|
|
|
|
writeName(_namespaces.getPrefix(namespaceURI), localName);
|
|
|
|
}
|
|
|
|
else if (namespaceURI.empty() && !qname.empty())
|
|
|
|
{
|
|
|
|
writeXML(qname);
|
|
|
|
}
|
|
|
|
else throw XMLException("Tag mismatch", nameToString(localName, qname));
|
|
|
|
|
|
|
|
CanonicalAttributeMap namespaceAttributeMap;
|
|
|
|
addNamespaceAttributes(namespaceAttributeMap);
|
|
|
|
writeAttributes(namespaceAttributeMap);
|
|
|
|
CanonicalAttributeMap attributeMap;
|
|
|
|
addAttributes(attributeMap, attributes, namespaceURI);
|
|
|
|
writeAttributes(attributeMap);
|
|
|
|
_unclosedStartTag = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
void XMLWriter::writeEndElement(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname)
|
|
|
|
{
|
|
|
|
if (_unclosedStartTag && !(_options & CANONICAL_XML))
|
|
|
|
{
|
|
|
|
writeMarkup(MARKUP_SLASHGT);
|
|
|
|
_unclosedStartTag = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (_unclosedStartTag)
|
|
|
|
{
|
|
|
|
writeMarkup(MARKUP_GT);
|
|
|
|
_unclosedStartTag = false;
|
|
|
|
}
|
|
|
|
writeMarkup(MARKUP_LTSLASH);
|
|
|
|
if (!localName.empty())
|
|
|
|
{
|
|
|
|
XMLString prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
writeName(prefix, localName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
writeXML(qname);
|
|
|
|
}
|
|
|
|
writeMarkup(MARKUP_GT);
|
|
|
|
}
|
|
|
|
_namespaces.popContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::closeStartTag()
|
|
|
|
{
|
|
|
|
_unclosedStartTag = false;
|
|
|
|
writeMarkup(MARKUP_GT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
void XMLWriter::declareNamespaces(const XMLString& namespaceURI, const XMLString& localName, const XMLString& qname, const Attributes& attributes)
|
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
std::map<XMLString, std::set<XMLString>> usedNamespaces;
|
2016-02-28 11:20:02 +01:00
|
|
|
bool defaultNameSpaceUsed = false;
|
2016-07-19 15:26:26 +02:00
|
|
|
XMLString defaultNamespaceURI = _namespaces.getURI(XMLString());
|
2016-02-28 11:20:02 +01:00
|
|
|
XMLString local;
|
|
|
|
XMLString prefix;
|
|
|
|
XMLString elementNamespaceURI = namespaceURI;
|
|
|
|
Name::split(qname, prefix, local);
|
|
|
|
if (elementNamespaceURI.empty())
|
|
|
|
elementNamespaceURI = _namespaces.getURI(prefix);
|
|
|
|
if (!elementNamespaceURI.empty())
|
|
|
|
{
|
|
|
|
usedNamespaces[prefix].insert(elementNamespaceURI);
|
|
|
|
if (!defaultNamespaceURI.empty() && elementNamespaceURI == defaultNamespaceURI)
|
|
|
|
defaultNameSpaceUsed = true;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < attributes.getLength(); i++)
|
|
|
|
{
|
|
|
|
XMLString attributeNamespaceURI = attributes.getURI(i);
|
2023-12-17 17:55:30 +02:00
|
|
|
const XMLString& attributeQName = attributes.getQName(i);
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
XMLString attributePrefix;
|
|
|
|
XMLString attributeLocal;
|
|
|
|
Name::split(attributeQName, attributePrefix, attributeLocal);
|
|
|
|
if (attributeNamespaceURI.empty())
|
|
|
|
attributeNamespaceURI = _namespaces.getURI(prefix);
|
|
|
|
if (!attributeNamespaceURI.empty())
|
|
|
|
{
|
|
|
|
usedNamespaces[attributePrefix].insert(attributeNamespaceURI);
|
|
|
|
defaultNameSpaceUsed = defaultNameSpaceUsed || (!defaultNamespaceURI.empty() && attributeNamespaceURI == defaultNamespaceURI);
|
|
|
|
}
|
|
|
|
}
|
2020-01-09 10:08:09 +01:00
|
|
|
for (const auto& p: usedNamespaces)
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
const std::set<XMLString> namespaceURIs = p.second;
|
|
|
|
for (const auto& nsURI: namespaceURIs)
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
XMLString prefix = p.first;
|
2022-07-07 17:18:20 +08:00
|
|
|
if (prefix.empty())
|
2020-01-09 10:08:09 +01:00
|
|
|
prefix = _namespaces.getPrefix(nsURI);
|
|
|
|
if (prefix.empty() && !_namespaces.isMapped(nsURI))
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
|
|
|
if (defaultNameSpaceUsed)
|
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
if (nsURI != defaultNamespaceURI)
|
2016-02-28 11:20:02 +01:00
|
|
|
prefix = uniquePrefix();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
defaultNamespaceURI = nsURI;
|
2016-02-28 11:20:02 +01:00
|
|
|
defaultNameSpaceUsed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
const XMLString& uri = _namespaces.getURI(prefix);
|
2022-07-07 17:18:20 +08:00
|
|
|
if ((uri.empty() || uri != nsURI) && !nsURI.empty())
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
_namespaces.declarePrefix(prefix, nsURI);
|
2016-02-28 11:20:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
void XMLWriter::declareAttributeNamespaces(const Attributes& attributes)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < attributes.getLength(); i++)
|
|
|
|
{
|
2023-12-17 17:55:30 +02:00
|
|
|
const XMLString& namespaceURI = attributes.getURI(i);
|
|
|
|
const XMLString& localName = attributes.getLocalName(i);
|
|
|
|
const XMLString& qname = attributes.getQName(i);
|
2012-04-29 18:52:25 +00:00
|
|
|
if (!localName.empty())
|
|
|
|
{
|
|
|
|
XMLString prefix;
|
|
|
|
XMLString splitLocalName;
|
|
|
|
Name::split(qname, prefix, splitLocalName);
|
|
|
|
if (prefix.empty()) prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
if (prefix.empty() && !namespaceURI.empty() && !_namespaces.isMapped(namespaceURI))
|
|
|
|
{
|
2012-11-10 11:49:39 +01:00
|
|
|
prefix = uniquePrefix();
|
2012-04-29 18:52:25 +00:00
|
|
|
_namespaces.declarePrefix(prefix, namespaceURI);
|
|
|
|
}
|
|
|
|
|
|
|
|
const XMLString& uri = _namespaces.getURI(prefix);
|
|
|
|
if ((uri.empty() || uri != namespaceURI) && !namespaceURI.empty())
|
|
|
|
{
|
|
|
|
_namespaces.declarePrefix(prefix, namespaceURI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::addNamespaceAttributes(AttributeMap& attributeMap)
|
|
|
|
{
|
|
|
|
NamespaceSupport::PrefixSet prefixes;
|
|
|
|
_namespaces.getDeclaredPrefixes(prefixes);
|
2020-01-09 10:08:09 +01:00
|
|
|
for (const auto& prefix: prefixes)
|
2012-04-29 18:52:25 +00:00
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
XMLString uri = _namespaces.getURI(prefix);
|
|
|
|
XMLString qname = NamespaceSupport::XMLNS_NAMESPACE_PREFIX;
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
if (!prefix.empty())
|
|
|
|
{
|
|
|
|
qname.append(toXMLString(MARKUP_COLON));
|
|
|
|
qname.append(prefix);
|
|
|
|
}
|
|
|
|
attributeMap[qname] = uri;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
void XMLWriter::addNamespaceAttributes(CanonicalAttributeMap& attributeMap)
|
|
|
|
{
|
|
|
|
NamespaceSupport::PrefixSet prefixes;
|
|
|
|
_namespaces.getDeclaredPrefixes(prefixes);
|
2020-01-09 10:08:09 +01:00
|
|
|
for (const auto& prefix: prefixes)
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
XMLString uri = _namespaces.getURI(prefix);
|
|
|
|
XMLString qname = NamespaceSupport::XMLNS_NAMESPACE_PREFIX;
|
2022-07-07 17:18:20 +08:00
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
if (!prefix.empty())
|
|
|
|
{
|
|
|
|
qname.append(toXMLString(MARKUP_COLON));
|
|
|
|
qname.append(prefix);
|
|
|
|
}
|
|
|
|
attributeMap.insert(std::make_pair(qname, std::make_pair(qname, uri)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
void XMLWriter::addAttributes(AttributeMap& attributeMap, const Attributes& attributes, const XMLString& elementNamespaceURI)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < attributes.getLength(); i++)
|
|
|
|
{
|
2023-12-17 17:55:30 +02:00
|
|
|
const XMLString& namespaceURI = attributes.getURI(i);
|
|
|
|
const XMLString& localName = attributes.getLocalName(i);
|
2012-04-29 18:52:25 +00:00
|
|
|
XMLString qname = attributes.getQName(i);
|
|
|
|
if (!localName.empty())
|
|
|
|
{
|
|
|
|
XMLString prefix;
|
|
|
|
if (!namespaceURI.empty())
|
|
|
|
prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
if (!prefix.empty())
|
|
|
|
{
|
|
|
|
qname = prefix;
|
|
|
|
qname.append(toXMLString(MARKUP_COLON));
|
|
|
|
}
|
|
|
|
else qname.clear();
|
|
|
|
qname.append(localName);
|
|
|
|
}
|
|
|
|
attributeMap[qname] = attributes.getValue(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-28 11:20:02 +01:00
|
|
|
void XMLWriter::addAttributes(CanonicalAttributeMap& attributeMap, const Attributes& attributes, const XMLString& elementNamespaceURI)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < attributes.getLength(); i++)
|
|
|
|
{
|
2023-12-17 17:55:30 +02:00
|
|
|
const XMLString& namespaceURI = attributes.getURI(i);
|
|
|
|
const XMLString& localName = attributes.getLocalName(i);
|
2016-02-28 11:20:02 +01:00
|
|
|
XMLString qname = attributes.getQName(i);
|
|
|
|
XMLString fullQName = qname;
|
|
|
|
if (!localName.empty())
|
|
|
|
{
|
|
|
|
XMLString prefix;
|
|
|
|
if (!namespaceURI.empty())
|
|
|
|
{
|
|
|
|
prefix = _namespaces.getPrefix(namespaceURI);
|
|
|
|
fullQName = namespaceURI;
|
|
|
|
fullQName.append(toXMLString(MARKUP_COLON));
|
|
|
|
}
|
|
|
|
else fullQName.clear();
|
|
|
|
if (!prefix.empty())
|
|
|
|
{
|
|
|
|
qname = prefix;
|
|
|
|
qname.append(toXMLString(MARKUP_COLON));
|
|
|
|
}
|
|
|
|
else qname.clear();
|
|
|
|
qname.append(localName);
|
|
|
|
fullQName.append(localName);
|
|
|
|
}
|
|
|
|
attributeMap.insert(std::make_pair(fullQName, std::make_pair(qname, attributes.getValue(i))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
void XMLWriter::writeAttributes(const AttributeMap& attributeMap)
|
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
for (const auto& ap: attributeMap)
|
2012-04-29 18:52:25 +00:00
|
|
|
{
|
|
|
|
if ((_options & PRETTY_PRINT) && (_options & PRETTY_PRINT_ATTRIBUTES))
|
|
|
|
{
|
|
|
|
writeNewLine();
|
|
|
|
writeIndent(_depth + 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
writeMarkup(MARKUP_SPACE);
|
|
|
|
}
|
2020-01-09 10:08:09 +01:00
|
|
|
writeXML(ap.first);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeMarkup(MARKUP_EQQUOT);
|
2020-01-09 10:08:09 +01:00
|
|
|
for (auto c: ap.second)
|
2012-04-29 18:52:25 +00:00
|
|
|
{
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case '"': writeMarkup(MARKUP_QUOTENC); break;
|
|
|
|
case '&': writeMarkup(MARKUP_AMPENC); break;
|
|
|
|
case '<': writeMarkup(MARKUP_LTENC); break;
|
2016-02-28 11:20:02 +01:00
|
|
|
case '>': writeMarkup(MARKUP_GTENC); break;
|
|
|
|
case '\t': writeMarkup(MARKUP_TABENC); break;
|
|
|
|
case '\r': writeMarkup(MARKUP_CRENC); break;
|
|
|
|
case '\n': writeMarkup(MARKUP_LFENC); break;
|
|
|
|
default:
|
|
|
|
if (c >= 0 && c < 32)
|
|
|
|
throw XMLException("Invalid character token.");
|
2022-07-07 17:18:20 +08:00
|
|
|
else
|
2016-02-28 11:20:02 +01:00
|
|
|
writeXML(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
writeMarkup(MARKUP_QUOT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeAttributes(const CanonicalAttributeMap& attributeMap)
|
|
|
|
{
|
2020-01-09 10:08:09 +01:00
|
|
|
for (const auto& ap: attributeMap)
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
|
|
|
if ((_options & PRETTY_PRINT) && (_options & PRETTY_PRINT_ATTRIBUTES))
|
|
|
|
{
|
|
|
|
writeNewLine();
|
|
|
|
writeIndent(_depth + 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
writeMarkup(MARKUP_SPACE);
|
|
|
|
}
|
2020-01-09 10:08:09 +01:00
|
|
|
writeXML(ap.second.first);
|
2016-02-28 11:20:02 +01:00
|
|
|
writeMarkup(MARKUP_EQQUOT);
|
2020-01-09 10:08:09 +01:00
|
|
|
for (auto c: ap.second.second)
|
2016-02-28 11:20:02 +01:00
|
|
|
{
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case '"': writeMarkup(MARKUP_QUOTENC); break;
|
|
|
|
case '&': writeMarkup(MARKUP_AMPENC); break;
|
|
|
|
case '<': writeMarkup(MARKUP_LTENC); break;
|
2012-04-29 18:52:25 +00:00
|
|
|
case '>': writeMarkup(MARKUP_GTENC); break;
|
|
|
|
case '\t': writeMarkup(MARKUP_TABENC); break;
|
|
|
|
case '\r': writeMarkup(MARKUP_CRENC); break;
|
|
|
|
case '\n': writeMarkup(MARKUP_LFENC); break;
|
|
|
|
default:
|
|
|
|
if (c >= 0 && c < 32)
|
|
|
|
throw XMLException("Invalid character token.");
|
2022-07-07 17:18:20 +08:00
|
|
|
else
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
writeMarkup(MARKUP_QUOT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeMarkup(const std::string& str) const
|
|
|
|
{
|
|
|
|
#if defined(XML_UNICODE_WCHAR_T)
|
|
|
|
const XMLString xmlString = toXMLString(str);
|
|
|
|
writeXML(xmlString);
|
|
|
|
#else
|
|
|
|
_pTextConverter->write(str.data(), (int) str.size());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeXML(const XMLString& str) const
|
|
|
|
{
|
|
|
|
_pTextConverter->write((const char*) str.data(), (int) str.size()*sizeof(XMLChar));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeXML(XMLChar ch) const
|
|
|
|
{
|
|
|
|
_pTextConverter->write((const char*) &ch, sizeof(ch));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeName(const XMLString& prefix, const XMLString& localName)
|
|
|
|
{
|
|
|
|
if (prefix.empty())
|
|
|
|
{
|
|
|
|
writeXML(localName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-07 17:18:20 +08:00
|
|
|
writeXML(prefix);
|
|
|
|
writeMarkup(MARKUP_COLON);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeXML(localName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeNewLine() const
|
|
|
|
{
|
|
|
|
if (_options & PRETTY_PRINT)
|
|
|
|
writeMarkup(_newLine);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeIndent() const
|
|
|
|
{
|
|
|
|
writeIndent(_depth);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeIndent(int depth) const
|
|
|
|
{
|
|
|
|
for (int i = 0; i < depth; ++i)
|
|
|
|
writeMarkup(_indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeXMLDeclaration()
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("<?xml version=\"1.0\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
if (!_encoding.empty())
|
|
|
|
{
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup(" encoding=\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeMarkup(_encoding);
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("\""s);
|
2012-04-29 18:52:25 +00:00
|
|
|
}
|
2024-01-30 09:35:35 +01:00
|
|
|
writeMarkup("?>"s);
|
2012-04-29 18:52:25 +00:00
|
|
|
writeNewLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string XMLWriter::nameToString(const XMLString& localName, const XMLString& qname)
|
|
|
|
{
|
|
|
|
if (qname.empty())
|
|
|
|
return fromXMLString(localName);
|
|
|
|
else
|
|
|
|
return fromXMLString(qname);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-10 11:49:39 +01:00
|
|
|
XMLString XMLWriter::uniquePrefix()
|
2012-04-29 18:52:25 +00:00
|
|
|
{
|
|
|
|
std::ostringstream str;
|
|
|
|
str << "ns" << ++_prefix;
|
|
|
|
return toXMLString(str.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-03 22:50:07 -05:00
|
|
|
bool XMLWriter::isNamespaceMapped(const XMLString& namespc) const
|
2012-11-10 11:49:39 +01:00
|
|
|
{
|
|
|
|
return _namespaces.isMapped(namespc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-29 18:52:25 +00:00
|
|
|
} } // namespace Poco::XML
|