#include <assert.h> #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 << string(depth * 2, ' '); } void XMLWriter::openElement(const string & name, const XMLAttrs & attrs) { assert(!closed); indent_(pendingElems.size()); output << "<" << name; writeAttrs(attrs); output << ">"; if (indent) output << std::endl; pendingElems.push_back(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(const string & 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 << "\""; } } }