/* */ #include "XML2SAXMetalinkProcessor.h" #include "BinaryStream.h" #include "MetalinkParserStateMachine.h" #include "Metalinker.h" #include "MetalinkEntry.h" #include "util.h" #include "message.h" #include "DlAbortEx.h" namespace aria2 { class SessionData { public: SharedHandle _stm; std::deque _charactersStack; SessionData(const SharedHandle& stm):_stm(stm) {} }; static void mlStartElement(void* userData, const xmlChar* name, const xmlChar** attrs) { SessionData* sd = reinterpret_cast(userData); std::map attrmap; if(attrs) { const xmlChar** p = attrs; while(*p != 0) { std::string name = (const char*)*p++; if(*p == 0) { break; } std::string value = util::trim((const char*)*p++); attrmap[name] = value; } } sd->_stm->beginElement((const char*)name, attrmap); if(sd->_stm->needsCharactersBuffering()) { sd->_charactersStack.push_front(std::string()); } } static void mlEndElement(void* userData, const xmlChar* name) { SessionData* sd = reinterpret_cast(userData); std::string characters; if(sd->_stm->needsCharactersBuffering()) { characters = util::trim(sd->_charactersStack.front()); sd->_charactersStack.pop_front(); } sd->_stm->endElement((const char*)name, characters); } static void mlCharacters(void* userData, const xmlChar* ch, int len) { SessionData* sd = reinterpret_cast(userData); if(sd->_stm->needsCharactersBuffering()) { sd->_charactersStack.front() += std::string(&ch[0], &ch[len]); } } static 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 &mlStartElement, // startElementSAXFunc &mlEndElement, // endElementSAXFunc 0, // referenceSAXFunc &mlCharacters, // charactersSAXFunc 0, // ignorableWhitespaceSAXFunc 0, // processingInstructionSAXFunc 0, // commentSAXFunc 0, // warningSAXFunc 0, // errorSAXFunc 0, // fatalErrorSAXFunc 0, // getParameterEntitySAXFunc 0, // cdataBlockSAXFunc 0, // externalSubsetSAXFunc 0, // unsigned int initialized 0, // void * _private 0, // startElementNsSAX2Func 0, // endElementNsSAX2Func 0, // xmlStructuredErrorFunc }; SharedHandle MetalinkProcessor::parseFile(const std::string& filename) { _stm.reset(new MetalinkParserStateMachine()); SharedHandle sessionData(new SessionData(_stm)); int retval = xmlSAXUserParseFile(&mySAXHandler, sessionData.get(), filename.c_str()); if(retval != 0) { throw DL_ABORT_EX(MSG_CANNOT_PARSE_METALINK); } return _stm->getResult(); } SharedHandle MetalinkProcessor::parseFromBinaryStream(const SharedHandle& binaryStream) { _stm.reset(new MetalinkParserStateMachine()); size_t bufSize = 4096; unsigned char buf[bufSize]; ssize_t res = binaryStream->readData(buf, 4, 0); if(res != 4) { throw DL_ABORT_EX("Too small data for parsing XML."); } SharedHandle sessionData(new SessionData(_stm)); xmlParserCtxtPtr ctx = xmlCreatePushParserCtxt(&mySAXHandler, sessionData.get(), (const char*)buf, res, 0); try { off_t readOffset = res; while(1) { ssize_t res = binaryStream->readData(buf, bufSize, readOffset); if(res == 0) { break; } if(xmlParseChunk(ctx, (const char*)buf, res, 0) != 0) { throw DL_ABORT_EX(MSG_CANNOT_PARSE_METALINK); } readOffset += res; } xmlParseChunk(ctx, (const char*)buf, 0, 1); } catch(Exception& e) { xmlFreeParserCtxt(ctx); throw; } xmlFreeParserCtxt(ctx); if(!_stm->finished()) { throw DL_ABORT_EX(MSG_CANNOT_PARSE_METALINK); } return _stm->getResult(); } } // namespace aria2