/**
* Redistribution and use of this software and associated documentation
* ("Software"), with or without modification, are permitted provided
* that the following conditions are met:
*
* 1. Redistributions of source code must retain copyright
* statements and notices. Redistributions must also contain a
* copy of this document.
*
* 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 "Exolab" must not be used to endorse or promote
* products derived from this Software without prior written
* permission of Intalio, Inc. For written permission,
* please contact info@exolab.org.
*
* 4. Products derived from this Software may not be called "Exolab"
* nor may "Exolab" appear in their names without prior written
* permission of Intalio, Inc. Exolab is a registered
* trademark of Intalio, Inc.
*
* 5. Due credit should be given to the Exolab Project
* (http://www.exolab.org/).
*
* THIS SOFTWARE IS PROVIDED BY INTALIO, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESSED 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
* INTALIO, INC. OR ITS CONTRIBUTORS 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.
*
* Copyright 2001 (C) Intalio, Inc. All Rights Reserved.
*
* $Id: SAX2ANY.java,v 1.1.1.1 2003/03/03 07:09:48 kvisco Exp $
* Date Author Changes
* 04/06/2001 Arnaud Blandin Created
*/
package org.exolab.castor.xml.util;
import org.exolab.castor.types.AnyNode;
import org.xml.sax.ContentHandler;
import org.xml.sax.DocumentHandler;
import org.xml.sax.ErrorHandler;
import org.xml.sax.Locator;
import org.xml.sax.Attributes;
import org.xml.sax.AttributeList;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.exolab.castor.util.Stack;
import org.exolab.castor.xml.Namespaces;
/**
* This class is a SAX Content Handler that
* build an AnyNode from a stream of SAX events (either SAX1 for compatibility or SAX2)
* @author <a href="blandin@intalio.com>Arnaud Blandin</a>
* @version $Revision: 1.1.1.1 $ $Date: 2003/03/03 07:09:48 $
*/
public class SAX2ANY implements ContentHandler, DocumentHandler, ErrorHandler
{
/**
* Prefix used by namespace declaration.
*/
private final static String XMLNS_PREFIX = "xmlns";
private final static int XMLNS_PREFIX_LENGTH = XMLNS_PREFIX.length() + 1; // prefix + ':'
private final static String XMLNS_NAMESPACE = "http://www.w3.org/2000/xmlns/";
/**
* The starting node.
*/
private AnyNode _startingNode;
/**
* The current AnyNode we are building
*/
private AnyNode _node;
/**
* A stack to store all the nodes we are creating
*/
private Stack _nodeStack = new Stack();
/**
* A stack to store the namespaces declaration
*/
private Stack _namespaces = new Stack();
/**
* The locator
*/
private Locator _locator;
/**
* A flag that indicates we are in a character section.
*/
private boolean _character = false;
/**
* The namespace context of this handler
*/
private Namespaces _context;
/**
* Default constructor
*/
public SAX2ANY() {
super();
init();
}
/**
* Constructs a SAX2ANY given a namespace context.
*
* @param context the namespace context in which this handler acts.
*/
public SAX2ANY(Namespaces context) {
_context = context;
init();
}
private void init() {
if (_context == null)
_context = new Namespaces();
}
/**
* Sets the document locator of the current parsed inputsource
* @param locator the Locator of the current parsed inputsource
*/
public void setDocumentLocator(Locator locator) {
_locator = locator;
}
//----------------- NOT IMPLEMENTED --------------
//we don't need to implement these methods since
//we are only dealing with xml fragments
public void startDocument() throws SAXException {
}
public void endDocument() throws SAXException {
}
public void ignorableWhitespace(char[] ch, int start, int length) throws SAXException {
return;
}
public void processingInstruction(String target, String data) throws SAXException {
return;
}
public void skippedEntity(String name) throws SAXException {
return;
}
//-------------------------------------------------
//--Namespace related (SAX2 Events)
public void startPrefixMapping(String prefix, String uri) throws SAXException {
AnyNode temp = new AnyNode(AnyNode.NAMESPACE, null, prefix, uri, null);
_namespaces.push(temp);
}
public void endPrefixMapping(String prefix) throws SAXException {
//nothing to do?
}
//--startElement methods SAX1 and SAX2
/**
* Implementation of {@link org.xml.sax.DocumentHandler#startElement}
*/
public void startElement(String name, AttributeList atts)
throws SAXException
{
_character = false;
String qName;
String value;
AnyNode tempNode = null;
//Namespace handling code to be moved once we integrate
//the new event API
/////////////////NAMESPACE HANDLING/////////////////////
_context = _context.createNamespaces();
String prefix = "";
String namespaceURI = null;
int idx = name.indexOf(':');
if (idx >= 0) {
prefix = name.substring(0,idx);
}
namespaceURI = _context.getNamespaceURI(prefix);
//--Overhead here since we process attributes twice
for (int i=0; i<atts.getLength(); ++i) {
qName = atts.getName(i);
value = atts.getValue(i);
String nsPrefix = null;
if (qName.startsWith(XMLNS_PREFIX)) {
//handles namespace declaration
// Extract the prefix if any
nsPrefix = (qName.equals(XMLNS_PREFIX))?null:qName.substring(XMLNS_PREFIX_LENGTH);
tempNode = new AnyNode(AnyNode.NAMESPACE, getLocalPart(qName), nsPrefix, value, null);
_context.addNamespace(nsPrefix, value);
_namespaces.push(tempNode);
if (prefix.equals(nsPrefix))
namespaceURI = value;
}
}
////////////////////////END OF NAMESPACE HANDLING///////////////
createNodeElement(namespaceURI, getLocalPart(name), name);
while (!_namespaces.empty()) {
tempNode = (AnyNode)_namespaces.pop();
_node.addNamespace(tempNode);
}
//process attributes
for (int i=0; i<atts.getLength(); ++i) {
qName = atts.getName(i);
value = atts.getValue(i);
//Namespace handling already done
if (!qName.startsWith(XMLNS_PREFIX)) {
tempNode = new AnyNode(AnyNode.ATTRIBUTE, getLocalPart(qName), null, null, value);
_node.addAttribute(tempNode);
}
}
tempNode = null;
}
/**
* Implementation of {@link org.xml.sax.ContentHandler#startElement}
*/
public void startElement(String namespaceURI, String localName,
String qName, Attributes atts)
throws SAXException
{
createNodeElement(namespaceURI, localName, qName);
//empty the namespaces stack
//that must have been filled by startPrefixMapping()
AnyNode tempNode = null;
while (!_namespaces.empty()) {
tempNode = (AnyNode)_namespaces.pop();
_node.addNamespace(tempNode);
}
//process attributes
for (int i=0; i<atts.getLength(); ++i) {
String uri = atts.getURI(i);
String qname = atts.getQName(i);
String localname = atts.getLocalName(i);
String value = atts.getValue(i);
tempNode = new AnyNode(AnyNode.ATTRIBUTE, localname, null, uri, value);
_node.addAttribute(tempNode);
}
tempNode = null;
}
//--endElement methods SAX1 and SAX2
public void endElement(String name)
throws SAXException
{
String prefix = "";
String namespaceURI = null;
int idx = name.indexOf(':');
if (idx >= 0) {
prefix = name.substring(0,idx);
}
namespaceURI = _context.getNamespaceURI(prefix);
endElement(namespaceURI,getLocalPart(name), name);
_context = _context.getParent();
}
public void endElement(String namespaceURI, String localName, String qName)
throws SAXException
{
_character = false;
//--if it is the starting element just returns
if (_startingNode.getLocalName().equals(localName) && _nodeStack.empty())
return;
//--else just add the node we have built to the previous node
_node = (AnyNode)_nodeStack.pop();
//-- if the stack is empty, we have a new child for the root node
//-- or a new sibling for the first child of the root node
if (_nodeStack.empty()) {
_startingNode.addChild(_node);
_node = _startingNode;
}
else {
AnyNode previousNode = (AnyNode) _nodeStack.peek();
previousNode.addChild(_node);
//--the node processing is finished -> come back to the previous node
_node = previousNode;
}
}
public void characters(char[] ch, int start, int length) throws SAXException {
//create a Text Node
String temp = new String(ch, start, length);
//skip whitespaces
if (isWhitespace(temp) & !_character)
return;
else {
AnyNode tempNode = new AnyNode(AnyNode.TEXT, null, null, null, temp);
_node.addChild(tempNode);
_character = true;
}
}
/**************************************************************************/
//implementation of ErrorHandler
public void warning(SAXParseException e) throws SAXException {
String err = "SAX2ANY warning\n"+
"Line : "+ e.getLineNumber() +'\n'+
"URI : " + e.getSystemId() +'\n'+ e.getMessage() ;
throw new SAXException (err, e);
} //warning
public void error(SAXParseException e) throws SAXException {
String err = "SAX2ANY Error \n"+
"Line : "+ e.getLineNumber() + '\n'+
"URI : " + e.getSystemId() +'\n'+ e.getMessage();
throw new SAXException (err,e);
} //error
public void fatalError(SAXParseException e) throws SAXException {
String err = "SAX2ANY Fatal Error \n"+
"Line : "+ e.getLineNumber() +'\n'+
"URI : " + e.getSystemId() +'\n'+ e.getMessage();
throw new SAXException (err,e);
} //fatalError
/*************************************************************************/
//Utility methods
public AnyNode getStartingNode() {
return _startingNode;
}
/**
* Get the namespace context of this SAX2ANY handler.
* If the SAX2ANY handler is called during the processing of an XML document, it
* may happen that the XML fragment parsed by the SAX2ANY handler contains
* references to namespaces declared in the given context.
*
* @return the namespace context to interact with while parsing an
* XML fragment with the SAX2ANY handler
*/
public Namespaces getNamespaceContext() {
return _context;
}
/**
* Set the namespace context in which this handler acts.
* If this handler is called during the processing of an XML document, it
* may happen that the XML fragment parsed by the SAX2ANY handler contains
* references to namespaces declared in the given context.
*
* @param context the namespace context to interact with while parsing an
* XML fragment with the SAX2ANY handler.
*/
public void setNamespaceContext(Namespaces context) {
_context = context;
}
/**
* Checks the given String to determine if it only
* contains whitespace.
*
* @param sb the String to check
* @return true if the only whitespace characters were
* found in the given StringBuffer
*/
private boolean isWhitespace(String string) {
for (int i = 0; i < string.length(); i++) {
char ch = string.charAt(i);
switch (ch) {
case ' ':
case '\n':
case '\t':
case '\r':
break;
default:
return false;
}
}
return true;
} //-- isWhitespace
/**
* Returns the local part of the given NCName. The local part is anything
* following the namespace prefix. If there is no namespace prefix
* the returned name will be the same as the given name.
* @return the local part of the given NCName.
*/
private String getLocalPart(String ncName) {
int idx = ncName.indexOf(':');
if (idx >= 0) return ncName.substring(idx+1);
return ncName;
} //-- getLocalPart
private void createNodeElement(String namespaceURI, String localName,
String qName)
{
String prefix = null;
//retrieves the prefix
if ((qName.length() != 0) && (qName.indexOf(':') != -1 ))
prefix = qName.substring(0,qName.indexOf(':'));
//creates the starting ELEMENT node
//or a default ELEMENT node
if ( (_nodeStack.empty()) && (_startingNode == null)) {
_startingNode = new AnyNode(AnyNode.ELEMENT, localName, prefix, namespaceURI, null);
_node = _startingNode;
} else {
_node = new AnyNode(AnyNode.ELEMENT, localName, prefix, namespaceURI, null);
//push the node in the stack
_nodeStack.push(_node);
}
}
}