97 lines
2 KiB
C++
97 lines
2 KiB
C++
#include <cassert>
|
|
|
|
#include "xml-writer.hh"
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
XMLWriter::XMLWriter(bool indent, std::ostream & output)
|
|
: output(output), indent(indent)
|
|
{
|
|
output << "<?xml version='1.0' encoding='utf-8'?>" << std::endl;
|
|
closed = false;
|
|
}
|
|
|
|
|
|
XMLWriter::~XMLWriter()
|
|
{
|
|
close();
|
|
}
|
|
|
|
|
|
void XMLWriter::close()
|
|
{
|
|
if (closed) return;
|
|
while (!pendingElems.empty()) closeElement();
|
|
closed = true;
|
|
}
|
|
|
|
|
|
void XMLWriter::indent_(size_t depth)
|
|
{
|
|
if (!indent) return;
|
|
output << std::string(depth * 2, ' ');
|
|
}
|
|
|
|
|
|
void XMLWriter::openElement(
|
|
std::string_view name,
|
|
const XMLAttrs & attrs)
|
|
{
|
|
assert(!closed);
|
|
indent_(pendingElems.size());
|
|
output << "<" << name;
|
|
writeAttrs(attrs);
|
|
output << ">";
|
|
if (indent) output << std::endl;
|
|
pendingElems.push_back(std::string(name));
|
|
}
|
|
|
|
|
|
void XMLWriter::closeElement()
|
|
{
|
|
assert(!pendingElems.empty());
|
|
indent_(pendingElems.size() - 1);
|
|
output << "</" << pendingElems.back() << ">";
|
|
if (indent) output << std::endl;
|
|
pendingElems.pop_back();
|
|
if (pendingElems.empty()) closed = true;
|
|
}
|
|
|
|
|
|
void XMLWriter::writeEmptyElement(
|
|
std::string_view name,
|
|
const XMLAttrs & attrs)
|
|
{
|
|
assert(!closed);
|
|
indent_(pendingElems.size());
|
|
output << "<" << name;
|
|
writeAttrs(attrs);
|
|
output << " />";
|
|
if (indent) output << std::endl;
|
|
}
|
|
|
|
|
|
void XMLWriter::writeAttrs(const XMLAttrs & attrs)
|
|
{
|
|
for (auto & i : attrs) {
|
|
output << " " << i.first << "=\"";
|
|
for (size_t j = 0; j < i.second.size(); ++j) {
|
|
char c = i.second[j];
|
|
if (c == '"') output << """;
|
|
else if (c == '<') output << "<";
|
|
else if (c == '>') output << ">";
|
|
else if (c == '&') output << "&";
|
|
/* Escape newlines to prevent attribute normalisation (see
|
|
XML spec, section 3.3.3. */
|
|
else if (c == '\n') output << "
";
|
|
else output << c;
|
|
}
|
|
output << "\"";
|
|
}
|
|
}
|
|
|
|
|
|
}
|