/* GnomeXMLReader.java - Copyright (C) 2004 Free Software Foundation, Inc. This file is part of GNU Classpath. GNU Classpath is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. GNU Classpath is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU Classpath; see the file COPYING. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. Linking this library statically or dynamically with other modules is making a combined work based on this library. Thus, the terms and conditions of the GNU General Public License cover the whole combination. As a special exception, the copyright holders of this library give you permission to link this library with independent modules to produce an executable, regardless of the license terms of these independent modules, and to copy and distribute the resulting executable under terms of your choice, provided that you also meet, for each linked independent module, the terms and conditions of the license of that module. An independent module is a module which is not derived from or based on this library. If you modify this library, you may extend this exception to your version of the library, but you are not obligated to do so. If you do not wish to do so, delete this exception statement from your version. */ package gnu.xml.libxmlj.sax; import java.io.File; import java.io.FileNotFoundException; import java.io.InputStream; import java.io.IOException; import java.net.MalformedURLException; import java.net.URL; import java.util.Arrays; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import org.xml.sax.Attributes; import org.xml.sax.ContentHandler; import org.xml.sax.DTDHandler; import org.xml.sax.EntityResolver; import org.xml.sax.ErrorHandler; import org.xml.sax.InputSource; import org.xml.sax.Locator; import org.xml.sax.SAXException; import org.xml.sax.SAXNotRecognizedException; import org.xml.sax.SAXNotSupportedException; import org.xml.sax.SAXParseException; import org.xml.sax.XMLReader; import org.xml.sax.ext.DeclHandler; import org.xml.sax.ext.LexicalHandler; import gnu.xml.libxmlj.util.NamedInputStream; import gnu.xml.libxmlj.util.StandaloneLocator; import gnu.xml.libxmlj.util.XMLJ; /** * A SAX2 parser that uses libxml2. * * @author <a href='mailto:dog@gnu.org'>Chris Burdess</a> */ public class GnomeXMLReader implements XMLReader { static { XMLJ.init (); } private static final String FEATURES_PREFIX = "http://xml.org/sax/features/"; private static final List RECOGNIZED_FEATURES = Arrays.asList (new String[] { "external-general-entities", "external-parameter-entities", "is-standalone", "lexical-handler/parameter-entities", "namespaces", "namespace-prefixes", "resolve-dtd-uris", "string-interning", "use-attributes2", "use-locator2", "use-entity-resolver2", "validation" }); private static final String PROPERTIES_PREFIX = "http://xml.org/sax/properties/"; private static final List RECOGNIZED_PROPERTIES = Arrays.asList (new String[] { "declaration-handler", "dom-node", "lexical-handler", "xml-string" }); // Features private transient boolean standalone; private boolean namespaces; private boolean namespacePrefixes; private boolean validation; // Callback handlers private ContentHandler contentHandler; private DTDHandler dtdHandler; private EntityResolver entityResolver; private ErrorHandler errorHandler; private DeclHandler declarationHandler; private LexicalHandler lexicalHandler; private GnomeLocator locator; // Namespace helper for handling callbacks private transient Namespaces ns; // If true, do not invoke callback methods except endDocument private transient boolean seenFatalError; private transient boolean seenStartDocument; private transient String base; public GnomeXMLReader () { this (true, true); } public GnomeXMLReader (boolean namespaces, boolean validation) { this.namespaces = namespaces; this.validation = validation; ns = new Namespaces (); } public ContentHandler getContentHandler () { return contentHandler; } public void setContentHandler (ContentHandler handler) { contentHandler = handler; } public DTDHandler getDTDHandler () { return dtdHandler; } public void setDTDHandler (DTDHandler handler) { dtdHandler = handler; } public EntityResolver getEntityResolver () { return entityResolver; } public void setEntityResolver (EntityResolver resolver) { entityResolver = resolver; } public ErrorHandler getErrorHandler () { return errorHandler; } public void setErrorHandler (ErrorHandler handler) { errorHandler = handler; } // Features public boolean getFeature (String name) throws SAXNotRecognizedException, SAXNotSupportedException { checkFeatureName (name); String key = name.substring (FEATURES_PREFIX.length ()); if ("external-general-entities".equals (key)) { return validation; // TODO check this } else if ("external-parameter-entities".equals (key)) { return validation; // TODO check this } else if ("is-standalone".equals (key)) { return standalone; } else if ("namespaces".equals (key)) { return namespaces; } else if ("namespace-prefixes".equals (key)) { return namespacePrefixes; } else if ("resolve-dtd-uris".equals (key)) { return true; } else if ("validation".equals (key)) { return validation; } else { return false; } } public void setFeature (String name, boolean value) throws SAXNotRecognizedException, SAXNotSupportedException { checkFeatureName (name); String key = name.substring (FEATURES_PREFIX.length ()); if ("namespaces".equals (key)) { namespaces = value; } else if ("namespace-prefixes".equals (key)) { namespacePrefixes = value; } else if ("validation".equals (key)) { validation = value; } } /** * Check that the specified feature name is recognized. */ static void checkFeatureName (String name) throws SAXNotRecognizedException { if (name == null || !name.startsWith (FEATURES_PREFIX)) { throw new SAXNotRecognizedException (name); } String key = name.substring (FEATURES_PREFIX.length ()); if (!RECOGNIZED_FEATURES.contains (key)) { throw new SAXNotRecognizedException (name); } } // Properties public Object getProperty (String name) throws SAXNotRecognizedException, SAXNotSupportedException { checkPropertyName (name); String key = name.substring (PROPERTIES_PREFIX.length ()); if ("declaration-handler".equals (key)) { return getDeclarationHandler (); } else if ("lexical-handler".equals (key)) { return getLexicalHandler (); } else { throw new SAXNotSupportedException (name); } } public void setProperty (String name, Object value) throws SAXNotRecognizedException, SAXNotSupportedException { checkPropertyName (name); String key = name.substring (PROPERTIES_PREFIX.length ()); if ("declaration-handler".equals (key)) { setDeclarationHandler ((DeclHandler) value); } else if ("lexical-handler".equals (key)) { setLexicalHandler ((LexicalHandler) value); } } public DeclHandler getDeclarationHandler () { return declarationHandler; } public void setDeclarationHandler (DeclHandler declarationHandler) { this.declarationHandler = declarationHandler; } public LexicalHandler getLexicalHandler () { return lexicalHandler; } public void setLexicalHandler (LexicalHandler lexicalHandler) { this.lexicalHandler = lexicalHandler; } /** * Check that the specified property name is recognized. */ static void checkPropertyName (String name) throws SAXNotRecognizedException { if (!name.startsWith (PROPERTIES_PREFIX)) { throw new SAXNotRecognizedException (name); } String key = name.substring (PROPERTIES_PREFIX.length ()); if (!RECOGNIZED_PROPERTIES.contains (key)) { throw new SAXNotRecognizedException (name); } } // Parse public void parse (String systemId) throws IOException, SAXException { URL url = null; try { url = new URL (systemId); } catch (MalformedURLException e) { File file = new File(systemId); if (!file.exists ()) { throw new FileNotFoundException (systemId); } String path = file.getAbsolutePath(); if (File.separatorChar != '/') { path = path.replace (File.separatorChar, '/'); } if (!path.startsWith ("/")) { path = "/" + path; } if (!path.endsWith ("/") && file.isDirectory ()) { path = path + "/"; } url = new URL ("file:" + path); } InputSource source = new InputSource(url.toString ()); source.setByteStream (url.openStream ()); parse (source); } public synchronized void parse (InputSource input) throws IOException, SAXException { NamedInputStream in = XMLJ.getInputStream (input); byte[] detectBuffer = in.getDetectBuffer (); String publicId = input.getPublicId (); String systemId = input.getSystemId (); base = XMLJ.getBaseURI (systemId); // Reset state standalone = false; seenFatalError = false; seenStartDocument = false; if (systemId != null) { int lsi = systemId.lastIndexOf ('/'); if (lsi != -1) { base = systemId.substring (0, lsi + 1); } } // Handle zero-length document if (detectBuffer == null) { startDocument (true); fatalError ("No document element", 0, 0, publicId, systemId); endDocument (); return; } // Parse parseStream(in, detectBuffer, publicId, systemId, base, validation, contentHandler != null, dtdHandler != null, entityResolver != null, errorHandler != null, declarationHandler != null, lexicalHandler != null); in.close (); } native void parseStream (InputStream in, byte[] detectBuffer, String publicId, String systemId, String base, boolean validate, boolean contentHandler, boolean dtdHandler, boolean entityResolver, boolean errorHandler, boolean declarationHandler, boolean lexicalHandler) throws IOException, SAXException; String getURI (String prefix) { if (!namespaces) { return null; } return ns.getURI (prefix); } // Callbacks from libxmlj private void startDTD (String name, String publicId, String systemId) throws SAXException { if (seenFatalError || lexicalHandler == null) { return; } try { systemId = XMLJ.getAbsoluteURI (base, systemId); lexicalHandler.startDTD (name, publicId, systemId); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void externalEntityDecl (String name, String publicId, String systemId) throws SAXException { if (seenFatalError || declarationHandler == null) { return; } try { systemId = XMLJ.getAbsoluteURI (base, systemId); declarationHandler.externalEntityDecl (name, publicId, systemId); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void internalEntityDecl (String name, String value) throws SAXException { if (seenFatalError || declarationHandler == null) { return; } try { declarationHandler.internalEntityDecl (name, value); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private InputStream resolveEntity (String publicId, String systemId) throws SAXException, IOException { if (entityResolver == null) { return null; } try { systemId = XMLJ.getAbsoluteURI (base, systemId); InputSource source = entityResolver.resolveEntity (publicId, systemId); return (source == null) ? null : XMLJ.getInputStream (source); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void notationDecl (String name, String publicId, String systemId) throws SAXException { if (seenFatalError || dtdHandler == null) { return; } try { systemId = XMLJ.getAbsoluteURI (base, systemId); dtdHandler.notationDecl (name, publicId, systemId); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void attributeDecl (String eName, String aName, String type, String mode, String value) throws SAXException { if (seenFatalError || declarationHandler == null) { return; } try { declarationHandler.attributeDecl (eName, aName, type, mode, value); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void elementDecl (String name, String model) throws SAXException { if (seenFatalError || declarationHandler == null) { return; } try { declarationHandler.elementDecl (name, model); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void unparsedEntityDecl (String name, String publicId, String systemId, String notationName) throws SAXException { if (seenFatalError || dtdHandler == null) { return; } try { systemId = XMLJ.getAbsoluteURI (base, systemId); dtdHandler.unparsedEntityDecl (name, publicId, systemId, notationName); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void setDocumentLocator (Object ctx, Object loc) { locator = new GnomeLocator (ctx, loc); if (seenFatalError || contentHandler == null) { return; } try { contentHandler.setDocumentLocator (locator); } catch (Exception e) { } } private void startDocument (boolean standalone) throws SAXException { this.standalone = standalone; seenStartDocument = true; if (contentHandler == null) { return; } try { contentHandler.startDocument (); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void endDocument () throws SAXException { if (contentHandler == null) { return; } try { contentHandler.endDocument(); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void startElement(String name, String[] attrs) throws SAXException { if (seenFatalError || contentHandler == null) { return; } try { XMLName xName = new XMLName (this, name); if (namespaces) { // Handle defined namespaces ns.push (); int len = (attrs == null) ? 0 : attrs.length; if (len > 0) { ArrayList filtered = new ArrayList (len); for (int i = 0; i < len; i += 2) { String attName = attrs[i]; String attValue = attrs[i + 1]; if (attName.equals ("xmlns")) { startPrefixMapping ("", attValue); } else if (attName.startsWith ("xmlns:")) { startPrefixMapping (attName.substring (6), attValue); } else { filtered.add (attName); filtered.add (attValue); } } // Remove xmlns attributes attrs = new String[filtered.size ()]; filtered.toArray (attrs); } } // Construct attributes Attributes atts = new StringArrayAttributes (this, attrs); contentHandler.startElement (xName.uri, xName.localName, xName.qName, atts); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void endElement (String name) throws SAXException { if (seenFatalError || contentHandler == null) { return; } try { XMLName xName = new XMLName (this, name); String uri = (xName.uri == null) ? "" : xName.uri; contentHandler.endElement (uri, xName.localName, xName.qName); // Handle undefining namespaces if (namespaces) { for (Iterator i = ns.currentPrefixes (); i.hasNext (); ) { endPrefixMapping ((String) i.next ()); } ns.pop (); // releases current depth } } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void startPrefixMapping (String prefix, String uri) throws SAXException { if (seenFatalError || contentHandler == null) { return; } ns.define (prefix, uri); contentHandler.startPrefixMapping (prefix, uri); } private void endPrefixMapping (String prefix) throws SAXException { if (seenFatalError || contentHandler == null) { return; } contentHandler.endPrefixMapping (prefix); } private void characters (String text) throws SAXException { if (seenFatalError || contentHandler == null || text == null) { return; } try { char[] ch = text.toCharArray (); contentHandler.characters (ch, 0, ch.length); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void ignorableWhitespace (String text) throws SAXException { if (seenFatalError || contentHandler == null || text == null) { return; } try { char[] ch = text.toCharArray (); contentHandler.ignorableWhitespace (ch, 0, ch.length); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void processingInstruction (String target, String data) throws SAXException { if (seenFatalError || contentHandler == null) { return; } try { if (data == null) { data = ""; } contentHandler.processingInstruction (target, data); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void comment (String text) throws SAXException { if (seenFatalError || lexicalHandler == null || text == null) { return; } try { char[] ch = text.toCharArray (); lexicalHandler.comment (ch, 0, ch.length); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void cdataBlock (String text) throws SAXException { if (seenFatalError || text == null) { return; } try { if (lexicalHandler == null) { characters(text); } else { lexicalHandler.startCDATA(); characters(text); lexicalHandler.endCDATA(); } } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void warning (String message, int lineNumber, int columnNumber, String publicId, String systemId) throws SAXException { if (seenFatalError || errorHandler == null) { return; } try { Locator l = new StandaloneLocator (lineNumber, columnNumber, publicId, systemId); errorHandler.warning (new SAXParseException (message, l)); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void error (String message, int lineNumber, int columnNumber, String publicId, String systemId) throws SAXException { if (seenFatalError || errorHandler == null) { return; } try { Locator l = new StandaloneLocator (lineNumber, columnNumber, publicId, systemId); errorHandler.error (new SAXParseException (message, l)); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } private void fatalError (String message, int lineNumber, int columnNumber, String publicId, String systemId) throws SAXException { if (seenFatalError || errorHandler == null) { return; } try { if (!seenStartDocument) { startDocument (false); } seenFatalError = true; Locator l = new StandaloneLocator (lineNumber, columnNumber, publicId, systemId); errorHandler.fatalError (new SAXParseException (message, l)); } catch (Exception e) { if (e instanceof SAXException) { throw (SAXException) e; } else { throw new SAXException (e); } } } }