/* * libjingle * Copyright 2004--2005, Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "talk/xmllite/xmlprinter.h" #include #include #include #include "talk/xmllite/xmlconstants.h" #include "talk/xmllite/xmlelement.h" #include "talk/xmllite/xmlnsstack.h" namespace buzz { class XmlPrinterImpl { public: XmlPrinterImpl(std::ostream* pout, XmlnsStack* ns_stack); void PrintElement(const XmlElement* element); void PrintQuotedValue(const std::string& text); void PrintBodyText(const std::string& text); void PrintCDATAText(const std::string& text); private: std::ostream *pout_; XmlnsStack* ns_stack_; }; void XmlPrinter::PrintXml(std::ostream* pout, const XmlElement* element) { XmlnsStack ns_stack; PrintXml(pout, element, &ns_stack); } void XmlPrinter::PrintXml(std::ostream* pout, const XmlElement* element, XmlnsStack* ns_stack) { XmlPrinterImpl printer(pout, ns_stack); printer.PrintElement(element); } XmlPrinterImpl::XmlPrinterImpl(std::ostream* pout, XmlnsStack* ns_stack) : pout_(pout), ns_stack_(ns_stack) { } void XmlPrinterImpl::PrintElement(const XmlElement* element) { ns_stack_->PushFrame(); // first go through attrs of pel to add xmlns definitions const XmlAttr* attr; for (attr = element->FirstAttr(); attr; attr = attr->NextAttr()) { if (attr->Name() == QN_XMLNS) { ns_stack_->AddXmlns(STR_EMPTY, attr->Value()); } else if (attr->Name().Namespace() == NS_XMLNS) { ns_stack_->AddXmlns(attr->Name().LocalPart(), attr->Value()); } } // then go through qnames to make sure needed xmlns definitons are added std::vector new_ns; std::pair prefix; prefix = ns_stack_->AddNewPrefix(element->Name().Namespace(), false); if (prefix.second) { new_ns.push_back(prefix.first); new_ns.push_back(element->Name().Namespace()); } for (attr = element->FirstAttr(); attr; attr = attr->NextAttr()) { prefix = ns_stack_->AddNewPrefix(attr->Name().Namespace(), true); if (prefix.second) { new_ns.push_back(prefix.first); new_ns.push_back(attr->Name().Namespace()); } } // print the element name *pout_ << '<' << ns_stack_->FormatQName(element->Name(), false); // and the attributes for (attr = element->FirstAttr(); attr; attr = attr->NextAttr()) { *pout_ << ' ' << ns_stack_->FormatQName(attr->Name(), true) << "=\""; PrintQuotedValue(attr->Value()); *pout_ << '"'; } // and the extra xmlns declarations std::vector::iterator i(new_ns.begin()); while (i < new_ns.end()) { if (*i == STR_EMPTY) { *pout_ << " xmlns=\"" << *(i + 1) << '"'; } else { *pout_ << " xmlns:" << *i << "=\"" << *(i + 1) << '"'; } i += 2; } // now the children const XmlChild* child = element->FirstChild(); if (child == NULL) *pout_ << "/>"; else { *pout_ << '>'; while (child) { if (child->IsText()) { if (element->IsCDATA()) { PrintCDATAText(child->AsText()->Text()); } else { PrintBodyText(child->AsText()->Text()); } } else { PrintElement(child->AsElement()); } child = child->NextChild(); } *pout_ << "FormatQName(element->Name(), false) << '>'; } ns_stack_->PopFrame(); } void XmlPrinterImpl::PrintQuotedValue(const std::string& text) { size_t safe = 0; for (;;) { size_t unsafe = text.find_first_of("<>&\"", safe); if (unsafe == std::string::npos) unsafe = text.length(); *pout_ << text.substr(safe, unsafe - safe); if (unsafe == text.length()) return; switch (text[unsafe]) { case '<': *pout_ << "<"; break; case '>': *pout_ << ">"; break; case '&': *pout_ << "&"; break; case '"': *pout_ << """; break; } safe = unsafe + 1; if (safe == text.length()) return; } } void XmlPrinterImpl::PrintBodyText(const std::string& text) { size_t safe = 0; for (;;) { size_t unsafe = text.find_first_of("<>&", safe); if (unsafe == std::string::npos) unsafe = text.length(); *pout_ << text.substr(safe, unsafe - safe); if (unsafe == text.length()) return; switch (text[unsafe]) { case '<': *pout_ << "<"; break; case '>': *pout_ << ">"; break; case '&': *pout_ << "&"; break; } safe = unsafe + 1; if (safe == text.length()) return; } } void XmlPrinterImpl::PrintCDATAText(const std::string& text) { *pout_ << ""; } } // namespace buzz