2006-08-03 15:21:21 +02:00
|
|
|
#include <assert.h>
|
|
|
|
|
2006-08-03 16:49:57 +02:00
|
|
|
#include "xml-writer.hh"
|
2006-08-03 15:21:21 +02:00
|
|
|
|
|
|
|
|
2006-09-04 23:06:23 +02:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::XMLWriter(bool indent, std::ostream & output)
|
2006-08-16 12:32:30 +02:00
|
|
|
: output(output), indent(indent)
|
2006-08-03 15:21:21 +02:00
|
|
|
{
|
2012-10-04 20:07:36 +02:00
|
|
|
output << "<?xml version='1.0' encoding='utf-8'?>" << std::endl;
|
2006-08-03 15:21:21 +02:00
|
|
|
closed = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XMLWriter::~XMLWriter()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::close()
|
|
|
|
{
|
|
|
|
if (closed) return;
|
|
|
|
while (!pendingElems.empty()) closeElement();
|
|
|
|
closed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-16 12:32:30 +02:00
|
|
|
void XMLWriter::indent_(unsigned int depth)
|
|
|
|
{
|
|
|
|
if (!indent) return;
|
|
|
|
output << string(depth * 2, ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-03 15:21:21 +02:00
|
|
|
void XMLWriter::openElement(const string & name,
|
|
|
|
const XMLAttrs & attrs)
|
|
|
|
{
|
|
|
|
assert(!closed);
|
2006-08-16 12:32:30 +02:00
|
|
|
indent_(pendingElems.size());
|
2006-08-03 15:21:21 +02:00
|
|
|
output << "<" << name;
|
|
|
|
writeAttrs(attrs);
|
|
|
|
output << ">";
|
2012-10-04 20:07:36 +02:00
|
|
|
if (indent) output << std::endl;
|
2006-08-03 15:21:21 +02:00
|
|
|
pendingElems.push_back(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::closeElement()
|
|
|
|
{
|
|
|
|
assert(!pendingElems.empty());
|
2006-08-16 12:32:30 +02:00
|
|
|
indent_(pendingElems.size() - 1);
|
2006-08-03 15:21:21 +02:00
|
|
|
output << "</" << pendingElems.back() << ">";
|
2012-10-04 20:07:36 +02:00
|
|
|
if (indent) output << std::endl;
|
2006-08-03 15:21:21 +02:00
|
|
|
pendingElems.pop_back();
|
|
|
|
if (pendingElems.empty()) closed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-03 17:52:09 +02:00
|
|
|
void XMLWriter::writeEmptyElement(const string & name,
|
2006-08-03 15:21:21 +02:00
|
|
|
const XMLAttrs & attrs)
|
|
|
|
{
|
|
|
|
assert(!closed);
|
2006-08-16 12:32:30 +02:00
|
|
|
indent_(pendingElems.size());
|
2006-08-03 15:21:21 +02:00
|
|
|
output << "<" << name;
|
|
|
|
writeAttrs(attrs);
|
|
|
|
output << " />";
|
2012-10-04 20:07:36 +02:00
|
|
|
if (indent) output << std::endl;
|
2006-08-03 15:21:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void XMLWriter::writeAttrs(const XMLAttrs & attrs)
|
|
|
|
{
|
2015-07-17 19:24:28 +02:00
|
|
|
for (auto & i : attrs) {
|
|
|
|
output << " " << i.first << "=\"";
|
|
|
|
for (unsigned int j = 0; j < i.second.size(); ++j) {
|
|
|
|
char c = i.second[j];
|
2006-08-03 15:21:21 +02:00
|
|
|
if (c == '"') output << """;
|
|
|
|
else if (c == '<') output << "<";
|
2010-03-31 10:29:01 +02:00
|
|
|
else if (c == '>') output << ">";
|
2006-08-03 15:21:21 +02:00
|
|
|
else if (c == '&') output << "&";
|
2006-08-24 17:02:39 +02:00
|
|
|
/* Escape newlines to prevent attribute normalisation (see
|
|
|
|
XML spec, section 3.3.3. */
|
|
|
|
else if (c == '\n') output << "
";
|
2006-08-03 15:21:21 +02:00
|
|
|
else output << c;
|
|
|
|
}
|
|
|
|
output << "\"";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-04 23:06:23 +02:00
|
|
|
}
|