/* */ #include "Xml2XmlParser.h" #include #include #include "a2io.h" #include "ParserStateMachine.h" #include "A2STR.h" #include "a2functional.h" #include "XmlAttr.h" namespace aria2 { namespace xml { namespace { void mlStartElement (void* userData, const xmlChar* localname, const xmlChar* prefix, const xmlChar* nsUri, int numNamespaces, const xmlChar** namespaces, int numAttrs, int numDefaulted, const xmlChar** attrs) { SessionData* sd = reinterpret_cast(userData); std::vector xmlAttrs; const char** pattrs = reinterpret_cast(attrs); for(size_t i = 0, max = numAttrs*5; i < max; i += 5) { XmlAttr xmlAttr; assert(pattrs[i]); xmlAttr.localname = pattrs[i]; if(pattrs[i+1]) { xmlAttr.prefix = pattrs[i+1]; } if(attrs[i+2]) { xmlAttr.nsUri = pattrs[i+2]; } xmlAttr.value = pattrs[i+3]; xmlAttr.valueLength = pattrs[i+4]-xmlAttr.value; xmlAttrs.push_back(xmlAttr); } sd->psm->beginElement (reinterpret_cast(localname), reinterpret_cast(prefix), reinterpret_cast(nsUri), xmlAttrs); if(sd->psm->needsCharactersBuffering()) { sd->charactersStack.push_front(A2STR::NIL); } } } // namespace namespace { void mlEndElement (void* userData, const xmlChar* localname, const xmlChar* prefix, const xmlChar* nsUri) { SessionData* sd = reinterpret_cast(userData); std::string characters; if(sd->psm->needsCharactersBuffering()) { characters = sd->charactersStack.front(); sd->charactersStack.pop_front(); } sd->psm->endElement (reinterpret_cast(localname), reinterpret_cast(prefix), reinterpret_cast(nsUri), characters); } } // namespace namespace { void mlCharacters(void* userData, const xmlChar* ch, int len) { SessionData* sd = reinterpret_cast(userData); if(sd->psm->needsCharactersBuffering()) { sd->charactersStack.front().append(&ch[0], &ch[len]); } } } // namespace namespace { xmlSAXHandler mySAXHandler = { 0, // internalSubsetSAXFunc 0, // isStandaloneSAXFunc 0, // hasInternalSubsetSAXFunc 0, // hasExternalSubsetSAXFunc 0, // resolveEntitySAXFunc 0, // getEntitySAXFunc 0, // entityDeclSAXFunc 0, // notationDeclSAXFunc 0, // attributeDeclSAXFunc 0, // elementDeclSAXFunc 0, // unparsedEntityDeclSAXFunc 0, // setDocumentLocatorSAXFunc 0, // startDocumentSAXFunc 0, // endDocumentSAXFunc 0, // startElementSAXFunc 0, // endElementSAXFunc 0, // referenceSAXFunc &mlCharacters, // charactersSAXFunc 0, // ignorableWhitespaceSAXFunc 0, // processingInstructionSAXFunc 0, // commentSAXFunc 0, // warningSAXFunc 0, // errorSAXFunc 0, // fatalErrorSAXFunc 0, // getParameterEntitySAXFunc 0, // cdataBlockSAXFunc 0, // externalSubsetSAXFunc XML_SAX2_MAGIC, // unsigned int initialized 0, // void * _private &mlStartElement, // startElementNsSAX2Func &mlEndElement, // endElementNsSAX2Func 0, // xmlStructuredErrorFunc }; } // namespace XmlParser::XmlParser(ParserStateMachine* psm) : psm_(psm), sessionData_(psm), ctx_(xmlCreatePushParserCtxt(&mySAXHandler, &sessionData_, 0, 0, 0)), lastError_(0) {} XmlParser::~XmlParser() { xmlFreeParserCtxt(ctx_); } ssize_t XmlParser::parseUpdate(const char* data, size_t size) { if(lastError_ != 0) { return lastError_; } int rv = xmlParseChunk(ctx_, data, size, 0); if(rv != 0) { return lastError_ = ERR_XML_PARSE; } else { return size; } } ssize_t XmlParser::parseFinal(const char* data, size_t size) { if(lastError_ != 0) { return lastError_; } int rv = xmlParseChunk(ctx_, data, size, 1); if(rv != 0) { return lastError_ = ERR_XML_PARSE; } else { return size; } } int XmlParser::reset() { psm_->reset(); sessionData_.reset(); int rv = xmlCtxtResetPush(ctx_, 0, 0, 0, 0); if(rv != 0) { return lastError_ = ERR_RESET; } else { return 0; } } } // namespace xml } // namespace aria2