251 lines
7.1 KiB
C++
251 lines
7.1 KiB
C++
/*
|
|
* The contents of this file are subject to the Mozilla Public
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is TransforMiiX XSLT processor.
|
|
*
|
|
* The Initial Developer of the Original Code is The MITRE Corporation.
|
|
* Portions created by MITRE are Copyright (C) 1999 The MITRE Corporation.
|
|
*
|
|
* Portions created by Keith Visco as a Non MITRE employee,
|
|
* (C) 1999 Keith Visco. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Tom Kneeland, tomk@mitre.org
|
|
* -- original author.
|
|
*
|
|
* Keith Visco, kvisco@ziplink.net
|
|
* -- finished implementation. Too many changes to list here.
|
|
*
|
|
* Bob Miller, Oblix Inc., kbob@oblix.com
|
|
* -- fixed assignment to "true" to be MB_TRUE, in ::parse()
|
|
*
|
|
* Marina Mechtcheriakova, mmarina@mindspring.com
|
|
* -- UNICODE fix in method startElement, changed casting of
|
|
* char* to DOM_CHAR* to use the proper String constructor,
|
|
* see method startElement
|
|
* -- Removed a number of castings of XML_Char to DOM_CHAR since they
|
|
* were not working on Windows properly
|
|
*
|
|
*/
|
|
|
|
#include "XMLParser.h"
|
|
#ifndef TX_EXE
|
|
#include "nsSyncLoader.h"
|
|
#include "URIUtils.h"
|
|
#include "nsIIOService.h"
|
|
#include "nsIURL.h"
|
|
#include "nsNetCID.h"
|
|
#include "nsIServiceManager.h"
|
|
#endif
|
|
/**
|
|
* Implementation of an In-Memory DOM based XML parser. The actual XML
|
|
* parsing is provided by EXPAT.
|
|
*
|
|
* @author <a href="kvisco@ziplink.net">Keith Visco</a>
|
|
* @author <a href="tomk@mitre.org">Tom Kneeland</a>
|
|
* @author <a href="kbob@oblix.com">Bob Miller</a>
|
|
*
|
|
* Modification History:
|
|
* Who When What
|
|
* TK 05/03/99 Created
|
|
* KV 06/15/1999 Fixed bug in parse method which read from cin instead of
|
|
* the istream parameter.
|
|
* KV 06/15/1999 Changed #parse method to return a Document
|
|
* KV 06/17/1999 made a bunch of changes
|
|
*
|
|
**/
|
|
|
|
/**
|
|
* Creates a new XMLParser
|
|
**/
|
|
XMLParser::XMLParser()
|
|
{
|
|
#ifdef TX_EXE
|
|
errorState = MB_FALSE;
|
|
#endif
|
|
} //-- XMLParser
|
|
|
|
|
|
XMLParser::~XMLParser()
|
|
{
|
|
//-- clean up
|
|
} //-- ~XMLParser
|
|
|
|
Document* XMLParser::getDocumentFromURI
|
|
(const String& href, const String& baseUri, String& errMsg)
|
|
{
|
|
|
|
String documentURL;
|
|
URIUtils::resolveHref(href, baseUri, documentURL);
|
|
|
|
#ifndef TX_EXE
|
|
nsresult rv = NS_OK;
|
|
nsCOMPtr<nsIURI> documentURI;
|
|
nsCOMPtr<nsIIOService> pService(do_GetService(NS_IOSERVICE_CONTRACTID,
|
|
&rv));
|
|
if (NS_FAILED(rv)) return NULL;
|
|
|
|
char *hrefStr = (documentURL.getConstNSString()).ToNewCString();
|
|
rv = pService->NewURI(hrefStr, nsnull, getter_AddRefs(documentURI));
|
|
nsCRT::free(hrefStr);
|
|
if (NS_FAILED(rv)) return NULL;
|
|
|
|
nsCOMPtr<nsISyncLoader>aLoader = do_CreateInstance( TRANSFORMIIX_SYNCLOADER_CONTRACTID, &rv );
|
|
if (NS_FAILED(rv)) return NULL;
|
|
|
|
nsCOMPtr <nsIDOMDocument> theDocument;
|
|
rv = aLoader->LoadDocument(documentURI, getter_AddRefs(theDocument));
|
|
if (NS_FAILED(rv)) return NULL;
|
|
|
|
return new Document(theDocument);
|
|
#else
|
|
istream* xslInput = URIUtils::getInputStream(documentURL, errMsg);
|
|
|
|
Document* resultDoc = 0;
|
|
if ( xslInput ) {
|
|
resultDoc = parse(*xslInput, documentURL);
|
|
delete xslInput;
|
|
}
|
|
if (!resultDoc) {
|
|
errMsg.append(getErrorString());
|
|
}
|
|
return resultDoc;
|
|
#endif
|
|
|
|
}
|
|
|
|
#ifdef TX_EXE
|
|
/**
|
|
* Parses the given input stream and returns a DOM Document.
|
|
* A NULL pointer will be returned if errors occurred
|
|
**/
|
|
Document* XMLParser::parse(istream& inputStream, const String& uri)
|
|
{
|
|
const int bufferSize = 1000;
|
|
|
|
char buf[bufferSize];
|
|
int done;
|
|
errorState = MB_FALSE;
|
|
errorString.clear();
|
|
if ( !inputStream ) {
|
|
errorString.append("unable to parse xml: invalid or unopen stream encountered.");
|
|
return NULL;
|
|
}
|
|
XML_Parser parser = XML_ParserCreate(NULL);
|
|
ParserState ps;
|
|
ps.document = new Document();
|
|
ps.document->documentBaseURI = uri;
|
|
ps.currentNode = ps.document;
|
|
|
|
XML_SetUserData(parser, &ps);
|
|
XML_SetElementHandler(parser, startElement, endElement);
|
|
XML_SetCharacterDataHandler(parser, charData);
|
|
XML_SetProcessingInstructionHandler(parser, piHandler);
|
|
XML_SetCommentHandler(parser,commentHandler);
|
|
do
|
|
{
|
|
inputStream.read(buf, bufferSize);
|
|
done = inputStream.eof();
|
|
|
|
if (!XML_Parse(parser, buf, inputStream.gcount(), done))
|
|
{
|
|
errorString.append(XML_ErrorString(XML_GetErrorCode(parser)));
|
|
errorString.append(" at line ");
|
|
errorString.append(XML_GetCurrentLineNumber(parser));
|
|
done = MB_TRUE;
|
|
errorState = MB_TRUE;
|
|
delete ps.document;
|
|
ps.document = NULL;
|
|
}
|
|
} while (!done);
|
|
inputStream.clear();
|
|
|
|
//if (currentElement)
|
|
//theDocument->appendChild(currentElement);
|
|
|
|
// clean up
|
|
XML_ParserFree(parser);
|
|
|
|
return ps.document;
|
|
}
|
|
|
|
const String& XMLParser::getErrorString()
|
|
{
|
|
return errorString;
|
|
}
|
|
|
|
|
|
void startElement(void *userData, const XML_Char *name, const XML_Char **atts)
|
|
{
|
|
ParserState* ps = (ParserState*)userData;
|
|
Element* newElement;
|
|
XML_Char* attName;
|
|
XML_Char* attValue;
|
|
XML_Char** theAtts = (XML_Char**)atts;
|
|
|
|
String nodeName((UNICODE_CHAR *)name);
|
|
newElement = ps->document->createElement(nodeName);
|
|
|
|
while (*theAtts)
|
|
{
|
|
attName = *theAtts++;
|
|
attValue = *theAtts++;
|
|
newElement->setAttribute((UNICODE_CHAR *)attName, (UNICODE_CHAR *)attValue);
|
|
}
|
|
|
|
ps->currentNode->appendChild(newElement);
|
|
ps->currentNode = newElement;
|
|
|
|
} //-- startElement
|
|
|
|
void endElement(void *userData, const XML_Char* name)
|
|
{
|
|
ParserState* ps = (ParserState*)userData;
|
|
if (ps->currentNode->getParentNode())
|
|
ps->currentNode = ps->currentNode->getParentNode();
|
|
} //-- endElement
|
|
|
|
void charData(void* userData, const XML_Char* s, int len)
|
|
{
|
|
ParserState* ps = (ParserState*)userData;
|
|
String data;
|
|
data.append((UNICODE_CHAR*)s, len);
|
|
Node* prevSib = ps->currentNode->getLastChild();
|
|
if (prevSib && prevSib->getNodeType()==Node::TEXT_NODE){
|
|
((CharacterData*)prevSib)->appendData(data);
|
|
} else {
|
|
ps->currentNode->appendChild(ps->document->createTextNode(data));
|
|
};
|
|
} //-- charData
|
|
|
|
void commentHandler(void* userData, const XML_Char* s)
|
|
{
|
|
ParserState* ps = (ParserState*)userData;
|
|
String data((UNICODE_CHAR*)s);
|
|
Node* prevSib = ps->currentNode->getLastChild();
|
|
ps->currentNode->appendChild(ps->document->createComment(data));
|
|
} //-- commentHandler
|
|
|
|
/**
|
|
* Handles ProcessingInstructions
|
|
**/
|
|
void piHandler(void *userData, const XML_Char *target, const XML_Char *data) {
|
|
ParserState* ps = (ParserState*)userData;
|
|
String targetStr((UNICODE_CHAR *)target);
|
|
String dataStr((UNICODE_CHAR *)data);
|
|
|
|
ps->currentNode->appendChild(
|
|
ps->document->createProcessingInstruction(targetStr, dataStr));
|
|
|
|
} //-- piHandler
|
|
|
|
#endif
|