12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610 |
- /* SAXDriver.java --
- Copyright (C) 1999,2000,2001,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.
- Portions derived from code which carried the following notice:
- Copyright (c) 1997, 1998 by Microstar Software Ltd.
- AElfred is free for both commercial and non-commercial use and
- redistribution, provided that Microstar's copyright and disclaimer are
- retained intact. You are free to modify AElfred for your own use and
- to redistribute AElfred with your modifications, provided that the
- modifications are clearly documented.
- This program 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. Please use it AT
- YOUR OWN RISK.
- */
- package gnu.xml.aelfred2;
- import java.io.*;
- import java.net.MalformedURLException;
- import java.net.URL;
- import java.util.Locale;
- import java.util.Stack;
- import java.util.ArrayList;
- import java.util.Collections;
- import java.util.Enumeration;
- import java.util.Iterator;
- import java.util.List;
- import org.xml.sax.*;
- import org.xml.sax.ext.*;
- import org.xml.sax.helpers.NamespaceSupport;
- /**
- * An enhanced SAX2 version of Microstar's Ælfred XML parser.
- * The enhancements primarily relate to significant improvements in
- * conformance to the XML specification, and SAX2 support. Performance
- * has been improved. See the package level documentation for more
- * information.
- *
- * <table border="1" width='100%' cellpadding='3' cellspacing='0'>
- * <tr bgcolor='#ccccff'>
- * <th><font size='+1'>Name</font></th>
- * <th><font size='+1'>Notes</font></th></tr>
- *
- * <tr><td colspan=2><center><em>Features ... URL prefix is
- * <b>http://xml.org/sax/features/</b></em></center></td></tr>
- *
- * <tr><td>(URL)/external-general-entities</td>
- * <td>Value defaults to <em>true</em></td></tr>
- * <tr><td>(URL)/external-parameter-entities</td>
- * <td>Value defaults to <em>true</em></td></tr>
- * <tr><td>(URL)/is-standalone</td>
- * <td>(PRELIMINARY) Returns true iff the document's parsing
- * has started (some non-error event after <em>startDocument()</em>
- * was reported) and the document's standalone flag is set.</td></tr>
- * <tr><td>(URL)/namespace-prefixes</td>
- * <td>Value defaults to <em>false</em> (but XML 1.0 names are
- * always reported)</td></tr>
- * <tr><td>(URL)/lexical-handler/parameter-entities</td>
- * <td>Value is fixed at <em>true</em></td></tr>
- * <tr><td>(URL)/namespaces</td>
- * <td>Value defaults to <em>true</em></td></tr>
- * <tr><td>(URL)/resolve-dtd-uris</td>
- * <td>(PRELIMINARY) Value defaults to <em>true</em></td></tr>
- * <tr><td>(URL)/string-interning</td>
- * <td>Value is fixed at <em>true</em></td></tr>
- * <tr><td>(URL)/use-attributes2</td>
- * <td>(PRELIMINARY) Value is fixed at <em>true</em></td></tr>
- * <tr><td>(URL)/use-entity-resolver2</td>
- * <td>(PRELIMINARY) Value defaults to <em>true</em></td></tr>
- * <tr><td>(URL)/validation</td>
- * <td>Value is fixed at <em>false</em></td></tr>
- *
- * <tr><td colspan=2><center><em>Handler Properties ... URL prefix is
- * <b>http://xml.org/sax/properties/</b></em></center></td></tr>
- *
- * <tr><td>(URL)/declaration-handler</td>
- * <td>A declaration handler may be provided. </td></tr>
- * <tr><td>(URL)/lexical-handler</td>
- * <td>A lexical handler may be provided. </td></tr>
- * </table>
- *
- * <p>This parser currently implements the SAX1 Parser API, but
- * it may not continue to do so in the future.
- *
- * @author Written by David Megginson (version 1.2a from Microstar)
- * @author Updated by David Brownell <dbrownell@users.sourceforge.net>
- * @see org.xml.sax.Parser
- */
- final public class SAXDriver
- implements Locator, Attributes2, XMLReader, Parser, AttributeList
- {
- private final DefaultHandler2 base = new DefaultHandler2();
- private XmlParser parser;
- private EntityResolver entityResolver = base;
- private EntityResolver2 resolver2 = null;
- private ContentHandler contentHandler = base;
- private DTDHandler dtdHandler = base;
- private ErrorHandler errorHandler = base;
- private DeclHandler declHandler = base;
- private LexicalHandler lexicalHandler = base;
- private String elementName;
- private Stack entityStack;
- // one vector (of object/struct): faster, smaller
- private List attributesList;
- private boolean namespaces = true;
- private boolean xmlNames = false;
- private boolean extGE = true;
- private boolean extPE = true;
- private boolean resolveAll = true;
- private boolean useResolver2 = true;
- // package private to allow (read-only) access in XmlParser
- boolean stringInterning = true;
- private int attributeCount;
- private boolean attributes;
- private String[] nsTemp;
- private NamespaceSupport prefixStack;
- //
- // Constructor.
- //
- /**
- * Constructs a SAX Parser.
- */
- public SAXDriver()
- {
- reset();
- }
- private void reset()
- {
- elementName = null;
- entityStack = new Stack();
- attributesList = Collections.synchronizedList(new ArrayList());
- attributeCount = 0;
- attributes = false;
- nsTemp = new String[3];
- prefixStack = null;
- }
- //
- // Implementation of org.xml.sax.Parser.
- //
- /**
- * <b>SAX1</b>: Sets the locale used for diagnostics; currently,
- * only locales using the English language are supported.
- * @param locale The locale for which diagnostics will be generated
- */
- public void setLocale(Locale locale)
- throws SAXException
- {
- if ("en".equals(locale.getLanguage()))
- {
- return;
- }
- throw new SAXException ("AElfred2 only supports English locales.");
- }
- /**
- * <b>SAX2</b>: Returns the object used when resolving external
- * entities during parsing (both general and parameter entities).
- */
- public EntityResolver getEntityResolver()
- {
- return (entityResolver == base) ? null : entityResolver;
- }
- /**
- * <b>SAX1, SAX2</b>: Set the entity resolver for this parser.
- * @param handler The object to receive entity events.
- */
- public void setEntityResolver(EntityResolver resolver)
- {
- if (resolver instanceof EntityResolver2)
- {
- resolver2 = (EntityResolver2) resolver;
- }
- else
- {
- resolver2 = null;
- }
- if (resolver == null)
- {
- resolver = base;
- }
- entityResolver = resolver;
- }
- /**
- * <b>SAX2</b>: Returns the object used to process declarations related
- * to notations and unparsed entities.
- */
- public DTDHandler getDTDHandler()
- {
- return (dtdHandler == base) ? null : dtdHandler;
- }
- /**
- * <b>SAX1, SAX2</b>: Set the DTD handler for this parser.
- * @param handler The object to receive DTD events.
- */
- public void setDTDHandler(DTDHandler handler)
- {
- if (handler == null)
- {
- handler = base;
- }
- this.dtdHandler = handler;
- }
- /**
- * <b>SAX1</b>: Set the document handler for this parser. If a
- * content handler was set, this document handler will supplant it.
- * The parser is set to report all XML 1.0 names rather than to
- * filter out "xmlns" attributes (the "namespace-prefixes" feature
- * is set to true).
- *
- * @deprecated SAX2 programs should use the XMLReader interface
- * and a ContentHandler.
- *
- * @param handler The object to receive document events.
- */
- public void setDocumentHandler(DocumentHandler handler)
- {
- contentHandler = new Adapter(handler);
- xmlNames = true;
- }
- /**
- * <b>SAX2</b>: Returns the object used to report the logical
- * content of an XML document.
- */
- public ContentHandler getContentHandler()
- {
- return (contentHandler == base) ? null : contentHandler;
- }
- /**
- * <b>SAX2</b>: Assigns the object used to report the logical
- * content of an XML document. If a document handler was set,
- * this content handler will supplant it (but XML 1.0 style name
- * reporting may remain enabled).
- */
- public void setContentHandler(ContentHandler handler)
- {
- if (handler == null)
- {
- handler = base;
- }
- contentHandler = handler;
- }
- /**
- * <b>SAX1, SAX2</b>: Set the error handler for this parser.
- * @param handler The object to receive error events.
- */
- public void setErrorHandler(ErrorHandler handler)
- {
- if (handler == null)
- {
- handler = base;
- }
- this.errorHandler = handler;
- }
- /**
- * <b>SAX2</b>: Returns the object used to receive callbacks for XML
- * errors of all levels (fatal, nonfatal, warning); this is never null;
- */
- public ErrorHandler getErrorHandler()
- {
- return (errorHandler == base) ? null : errorHandler;
- }
- /**
- * <b>SAX1, SAX2</b>: Auxiliary API to parse an XML document, used mostly
- * when no URI is available.
- * If you want anything useful to happen, you should set
- * at least one type of handler.
- * @param source The XML input source. Don't set 'encoding' unless
- * you know for a fact that it's correct.
- * @see #setEntityResolver
- * @see #setDTDHandler
- * @see #setContentHandler
- * @see #setErrorHandler
- * @exception SAXException The handlers may throw any SAXException,
- * and the parser normally throws SAXParseException objects.
- * @exception IOException IOExceptions are normally through through
- * the parser if there are problems reading the source document.
- */
- public void parse(InputSource source)
- throws SAXException, IOException
- {
- synchronized (base)
- {
- parser = new XmlParser();
- if (namespaces)
- {
- prefixStack = new NamespaceSupport();
- }
- else if (!xmlNames)
- {
- throw new IllegalStateException();
- }
- parser.setHandler(this);
- try
- {
- Reader r = source.getCharacterStream();
- InputStream in = source.getByteStream();
- parser.doParse(source.getSystemId(),
- source.getPublicId(),
- r,
- in,
- source.getEncoding());
- }
- catch (SAXException e)
- {
- throw e;
- }
- catch (IOException e)
- {
- throw e;
- }
- catch (RuntimeException e)
- {
- throw e;
- }
- catch (Exception e)
- {
- throw new SAXParseException(e.getMessage(), this, e);
- }
- finally
- {
- contentHandler.endDocument();
- reset();
- }
- }
- }
- /**
- * <b>SAX1, SAX2</b>: Preferred API to parse an XML document, using a
- * system identifier (URI).
- */
- public void parse(String systemId)
- throws SAXException, IOException
- {
- parse(new InputSource(systemId));
- }
- //
- // Implementation of SAX2 "XMLReader" interface
- //
- static final String FEATURE = "http://xml.org/sax/features/";
- static final String PROPERTY = "http://xml.org/sax/properties/";
- /**
- * <b>SAX2</b>: Tells the value of the specified feature flag.
- *
- * @exception SAXNotRecognizedException thrown if the feature flag
- * is neither built in, nor yet assigned.
- */
- public boolean getFeature(String featureId)
- throws SAXNotRecognizedException, SAXNotSupportedException
- {
- if ((FEATURE + "validation").equals(featureId))
- {
- return false;
- }
- // external entities (both types) are optionally included
- if ((FEATURE + "external-general-entities").equals(featureId))
- {
- return extGE;
- }
- if ((FEATURE + "external-parameter-entities").equals(featureId))
- {
- return extPE;
- }
- // element/attribute names are as written in document; no mangling
- if ((FEATURE + "namespace-prefixes").equals(featureId))
- {
- return xmlNames;
- }
- // report element/attribute namespaces?
- if ((FEATURE + "namespaces").equals(featureId))
- {
- return namespaces;
- }
- // all PEs and GEs are reported
- if ((FEATURE + "lexical-handler/parameter-entities").equals(featureId))
- {
- return true;
- }
- // default is true
- if ((FEATURE + "string-interning").equals(featureId))
- {
- return stringInterning;
- }
- // EXTENSIONS 1.1
- // always returns isSpecified info
- if ((FEATURE + "use-attributes2").equals(featureId))
- {
- return true;
- }
- // meaningful between startDocument/endDocument
- if ((FEATURE + "is-standalone").equals(featureId))
- {
- if (parser == null)
- {
- throw new SAXNotSupportedException(featureId);
- }
- return parser.isStandalone();
- }
- // optionally don't absolutize URIs in declarations
- if ((FEATURE + "resolve-dtd-uris").equals(featureId))
- {
- return resolveAll;
- }
- // optionally use resolver2 interface methods, if possible
- if ((FEATURE + "use-entity-resolver2").equals(featureId))
- {
- return useResolver2;
- }
- throw new SAXNotRecognizedException(featureId);
- }
- // package private
- DeclHandler getDeclHandler()
- {
- return declHandler;
- }
- // package private
- boolean resolveURIs()
- {
- return resolveAll;
- }
- /**
- * <b>SAX2</b>: Returns the specified property.
- *
- * @exception SAXNotRecognizedException thrown if the property value
- * is neither built in, nor yet stored.
- */
- public Object getProperty(String propertyId)
- throws SAXNotRecognizedException
- {
- if ((PROPERTY + "declaration-handler").equals(propertyId))
- {
- return (declHandler == base) ? null : declHandler;
- }
- if ((PROPERTY + "lexical-handler").equals(propertyId))
- {
- return (lexicalHandler == base) ? null : lexicalHandler;
- }
- // unknown properties
- throw new SAXNotRecognizedException(propertyId);
- }
- /**
- * <b>SAX2</b>: Sets the state of feature flags in this parser. Some
- * built-in feature flags are mutable.
- */
- public void setFeature(String featureId, boolean value)
- throws SAXNotRecognizedException, SAXNotSupportedException
- {
- boolean state;
- // Features with a defined value, we just change it if we can.
- state = getFeature (featureId);
- if (state == value)
- {
- return;
- }
- if (parser != null)
- {
- throw new SAXNotSupportedException("not while parsing");
- }
- if ((FEATURE + "namespace-prefixes").equals(featureId))
- {
- // in this implementation, this only affects xmlns reporting
- xmlNames = value;
- // forcibly prevent illegal parser state
- if (!xmlNames)
- {
- namespaces = true;
- }
- return;
- }
- if ((FEATURE + "namespaces").equals(featureId))
- {
- namespaces = value;
- // forcibly prevent illegal parser state
- if (!namespaces)
- {
- xmlNames = true;
- }
- return;
- }
- if ((FEATURE + "external-general-entities").equals(featureId))
- {
- extGE = value;
- return;
- }
- if ((FEATURE + "external-parameter-entities").equals(featureId))
- {
- extPE = value;
- return;
- }
- if ((FEATURE + "resolve-dtd-uris").equals(featureId))
- {
- resolveAll = value;
- return;
- }
- if ((FEATURE + "use-entity-resolver2").equals(featureId))
- {
- useResolver2 = value;
- return;
- }
- throw new SAXNotRecognizedException(featureId);
- }
- /**
- * <b>SAX2</b>: Assigns the specified property. Like SAX1 handlers,
- * these may be changed at any time.
- */
- public void setProperty(String propertyId, Object value)
- throws SAXNotRecognizedException, SAXNotSupportedException
- {
- // see if the property is recognized
- getProperty(propertyId);
- // Properties with a defined value, we just change it if we can.
- if ((PROPERTY + "declaration-handler").equals(propertyId))
- {
- if (value == null)
- {
- declHandler = base;
- }
- else if (!(value instanceof DeclHandler))
- {
- throw new SAXNotSupportedException(propertyId);
- }
- else
- {
- declHandler = (DeclHandler) value;
- }
- return ;
- }
- if ((PROPERTY + "lexical-handler").equals(propertyId))
- {
- if (value == null)
- {
- lexicalHandler = base;
- }
- else if (!(value instanceof LexicalHandler))
- {
- throw new SAXNotSupportedException(propertyId);
- }
- else
- {
- lexicalHandler = (LexicalHandler) value;
- }
- return;
- }
- throw new SAXNotSupportedException(propertyId);
- }
- //
- // This is where the driver receives XmlParser callbacks and translates
- // them into SAX callbacks. Some more callbacks have been added for
- // SAX2 support.
- //
- void startDocument()
- throws SAXException
- {
- contentHandler.setDocumentLocator(this);
- contentHandler.startDocument();
- attributesList.clear();
- }
- void skippedEntity(String name)
- throws SAXException
- {
- contentHandler.skippedEntity(name);
- }
- InputSource getExternalSubset(String name, String baseURI)
- throws SAXException, IOException
- {
- if (resolver2 == null || !useResolver2 || !extPE)
- {
- return null;
- }
- return resolver2.getExternalSubset(name, baseURI);
- }
- InputSource resolveEntity(boolean isPE, String name,
- InputSource in, String baseURI)
- throws SAXException, IOException
- {
- InputSource source;
- // external entities might be skipped
- if (isPE && !extPE)
- {
- return null;
- }
- if (!isPE && !extGE)
- {
- return null;
- }
- // ... or not
- lexicalHandler.startEntity(name);
- if (resolver2 != null && useResolver2)
- {
- source = resolver2.resolveEntity(name, in.getPublicId(),
- baseURI, in.getSystemId());
- if (source == null)
- {
- in.setSystemId(absolutize(baseURI,
- in.getSystemId(), false));
- source = in;
- }
- }
- else
- {
- in.setSystemId(absolutize(baseURI,
- in.getSystemId(),
- entityResolver != base));
- source = entityResolver.resolveEntity(in.getPublicId(),
- in.getSystemId());
- if (source == null)
- {
- source = in;
- }
- }
- startExternalEntity(name, source.getSystemId(), true);
- return source;
- }
- // absolutize a system ID relative to the specified base URI
- // (temporarily) package-visible for external entity decls
- String absolutize(String baseURI, String systemId, boolean nice)
- throws MalformedURLException, SAXException
- {
- // FIXME normalize system IDs -- when?
- // - Convert to UTF-8
- // - Map reserved and non-ASCII characters to %HH
- try
- {
- if (baseURI == null)
- {
- if (XmlParser.uriWarnings)
- {
- warn ("No base URI; hope this SYSTEM id is absolute: "
- + systemId);
- }
- return new URL(systemId).toString();
- }
- else
- {
- return new URL(new URL(baseURI), systemId).toString();
- }
- }
- catch (MalformedURLException e)
- {
- // Let unknown URI schemes pass through unless we need
- // the JVM to map them to i/o streams for us...
- if (!nice)
- {
- throw e;
- }
- // sometimes sysids for notations or unparsed entities
- // aren't really URIs...
- warn("Can't absolutize SYSTEM id: " + e.getMessage());
- return systemId;
- }
- }
- void startExternalEntity(String name, String systemId, boolean stackOnly)
- throws SAXException
- {
- // The following warning was deleted because the application has the
- // option of not setting systemId. Sun's JAXP or Xerces seems to
- // ignore this case.
- /*
- if (systemId == null)
- warn ("URI was not reported to parser for entity " + name);
- */
- if (!stackOnly) // spliced [dtd] needs startEntity
- {
- lexicalHandler.startEntity(name);
- }
- entityStack.push(systemId);
- }
- void endExternalEntity(String name)
- throws SAXException
- {
- if (!"[document]".equals(name))
- {
- lexicalHandler.endEntity(name);
- }
- entityStack.pop();
- }
- void startInternalEntity(String name)
- throws SAXException
- {
- lexicalHandler.startEntity(name);
- }
- void endInternalEntity(String name)
- throws SAXException
- {
- lexicalHandler.endEntity(name);
- }
- void doctypeDecl(String name, String publicId, String systemId)
- throws SAXException
- {
- lexicalHandler.startDTD(name, publicId, systemId);
- // ... the "name" is a declaration and should be given
- // to the DeclHandler (but sax2 doesn't).
- // the IDs for the external subset are lexical details,
- // as are the contents of the internal subset; but sax2
- // doesn't provide the internal subset "pre-parse"
- }
- void notationDecl(String name, String publicId, String systemId,
- String baseUri)
- throws SAXException
- {
- try
- {
- dtdHandler.notationDecl(name, publicId,
- (resolveAll && systemId != null)
- ? absolutize(baseUri, systemId, true)
- : systemId);
- }
- catch (IOException e)
- {
- // "can't happen"
- throw new SAXParseException(e.getMessage(), this, e);
- }
- }
- void unparsedEntityDecl(String name, String publicId, String systemId,
- String baseUri, String notation)
- throws SAXException
- {
- try
- {
- dtdHandler.unparsedEntityDecl(name, publicId,
- resolveAll
- ? absolutize(baseUri, systemId, true)
- : systemId,
- notation);
- }
- catch (IOException e)
- {
- // "can't happen"
- throw new SAXParseException(e.getMessage(), this, e);
- }
- }
- void endDoctype()
- throws SAXException
- {
- lexicalHandler.endDTD();
- }
- private void declarePrefix(String prefix, String uri)
- throws SAXException
- {
- int index = uri.indexOf(':');
- // many versions of nwalsh docbook stylesheets
- // have bogus URLs; so this can't be an error...
- if (index < 1 && uri.length() != 0)
- {
- warn("relative URI for namespace: " + uri);
- }
- // FIXME: char [0] must be ascii alpha; chars [1..index]
- // must be ascii alphanumeric or in "+-." [RFC 2396]
- //Namespace Constraints
- //name for xml prefix must be http://www.w3.org/XML/1998/namespace
- boolean prefixEquality = prefix.equals("xml");
- boolean uriEquality = uri.equals("http://www.w3.org/XML/1998/namespace");
- if ((prefixEquality || uriEquality) && !(prefixEquality && uriEquality))
- {
- fatal("xml is by definition bound to the namespace name " +
- "http://www.w3.org/XML/1998/namespace");
- }
- //xmlns prefix declaration is illegal but xml prefix declaration is llegal...
- if (prefixEquality && uriEquality)
- {
- return;
- }
- //name for xmlns prefix must be http://www.w3.org/2000/xmlns/
- prefixEquality = prefix.equals("xmlns");
- uriEquality = uri.equals("http://www.w3.org/2000/xmlns/");
- if ((prefixEquality || uriEquality) && !(prefixEquality && uriEquality))
- {
- fatal("http://www.w3.org/2000/xmlns/ is by definition bound" +
- " to prefix xmlns");
- }
- //even if the uri is http://www.w3.org/2000/xmlns/
- // it is illegal to declare it
- if (prefixEquality && uriEquality)
- {
- fatal ("declaring the xmlns prefix is illegal");
- }
- uri = uri.intern();
- prefixStack.declarePrefix(prefix, uri);
- contentHandler.startPrefixMapping(prefix, uri);
- }
- void attribute(String qname, String value, boolean isSpecified)
- throws SAXException
- {
- if (!attributes)
- {
- attributes = true;
- if (namespaces)
- {
- prefixStack.pushContext();
- }
- }
- // process namespace decls immediately;
- // then maybe forget this as an attribute
- if (namespaces)
- {
- int index;
- // default NS declaration?
- if (stringInterning)
- {
- if ("xmlns" == qname)
- {
- declarePrefix("", value);
- if (!xmlNames)
- {
- return;
- }
- }
- // NS prefix declaration?
- else if ((index = qname.indexOf(':')) == 5
- && qname.startsWith("xmlns"))
- {
- String prefix = qname.substring(6);
- if (prefix.equals(""))
- {
- fatal("missing prefix " +
- "in namespace declaration attribute");
- }
- if (value.length() == 0)
- {
- verror("missing URI in namespace declaration attribute: "
- + qname);
- }
- else
- {
- declarePrefix(prefix, value);
- }
- if (!xmlNames)
- {
- return;
- }
- }
- }
- else
- {
- if ("xmlns".equals(qname))
- {
- declarePrefix("", value);
- if (!xmlNames)
- {
- return;
- }
- }
- // NS prefix declaration?
- else if ((index = qname.indexOf(':')) == 5
- && qname.startsWith("xmlns"))
- {
- String prefix = qname.substring(6);
- if (value.length() == 0)
- {
- verror("missing URI in namespace decl attribute: "
- + qname);
- }
- else
- {
- declarePrefix(prefix, value);
- }
- if (!xmlNames)
- {
- return;
- }
- }
- }
- }
- // remember this attribute ...
- attributeCount++;
- // attribute type comes from querying parser's DTD records
- attributesList.add(new Attribute(qname, value, isSpecified));
- }
- void startElement(String elname)
- throws SAXException
- {
- ContentHandler handler = contentHandler;
- //
- // NOTE: this implementation of namespace support adds something
- // like six percent to parsing CPU time, in a large (~50 MB)
- // document that doesn't use namespaces at all. (Measured by PC
- // sampling, with a bug where endElement processing was omitted.)
- // [Measurement referred to older implementation, older JVM ...]
- //
- // It ought to become notably faster in such cases. Most
- // costs are the prefix stack calling Hashtable.get() (2%),
- // String.hashCode() (1.5%) and about 1.3% each for pushing
- // the context, and two chunks of name processing.
- //
- if (!attributes)
- {
- if (namespaces)
- {
- prefixStack.pushContext();
- }
- }
- else if (namespaces)
- {
- // now we can patch up namespace refs; we saw all the
- // declarations, so now we'll do the Right Thing
- Iterator itt = attributesList.iterator();
- while (itt.hasNext())
- {
- Attribute attribute = (Attribute) itt.next();
- String qname = attribute.name;
- int index;
- // default NS declaration?
- if (stringInterning)
- {
- if ("xmlns" == qname)
- {
- continue;
- }
- }
- else
- {
- if ("xmlns".equals(qname))
- {
- continue;
- }
- }
- //Illegal in the new Namespaces Draft
- //should it be only in 1.1 docs??
- if (qname.equals (":"))
- {
- fatal("namespace names consisting of a single colon " +
- "character are invalid");
- }
- index = qname.indexOf(':');
- // NS prefix declaration?
- if (index == 5 && qname.startsWith("xmlns"))
- {
- continue;
- }
- // it's not a NS decl; patch namespace info items
- if (prefixStack.processName(qname, nsTemp, true) == null)
- {
- fatal("undeclared attribute prefix in: " + qname);
- }
- else
- {
- attribute.nameSpace = nsTemp[0];
- attribute.localName = nsTemp[1];
- }
- }
- }
- // save element name so attribute callbacks work
- elementName = elname;
- if (namespaces)
- {
- if (prefixStack.processName(elname, nsTemp, false) == null)
- {
- fatal("undeclared element prefix in: " + elname);
- nsTemp[0] = nsTemp[1] = "";
- }
- handler.startElement(nsTemp[0], nsTemp[1], elname, this);
- }
- else
- {
- handler.startElement("", "", elname, this);
- }
- // elementName = null;
- // elements with no attributes are pretty common!
- if (attributes)
- {
- attributesList.clear();
- attributeCount = 0;
- attributes = false;
- }
- }
- void endElement(String elname)
- throws SAXException
- {
- ContentHandler handler = contentHandler;
- if (!namespaces)
- {
- handler.endElement("", "", elname);
- return;
- }
- prefixStack.processName(elname, nsTemp, false);
- handler.endElement(nsTemp[0], nsTemp[1], elname);
- Enumeration prefixes = prefixStack.getDeclaredPrefixes();
- while (prefixes.hasMoreElements())
- {
- handler.endPrefixMapping((String) prefixes.nextElement());
- }
- prefixStack.popContext();
- }
- void startCDATA()
- throws SAXException
- {
- lexicalHandler.startCDATA();
- }
- void charData(char[] ch, int start, int length)
- throws SAXException
- {
- contentHandler.characters(ch, start, length);
- }
- void endCDATA()
- throws SAXException
- {
- lexicalHandler.endCDATA();
- }
- void ignorableWhitespace(char[] ch, int start, int length)
- throws SAXException
- {
- contentHandler.ignorableWhitespace(ch, start, length);
- }
- void processingInstruction(String target, String data)
- throws SAXException
- {
- contentHandler.processingInstruction(target, data);
- }
- void comment(char[] ch, int start, int length)
- throws SAXException
- {
- if (lexicalHandler != base)
- {
- lexicalHandler.comment(ch, start, length);
- }
- }
- void fatal(String message)
- throws SAXException
- {
- SAXParseException fatal;
- fatal = new SAXParseException(message, this);
- errorHandler.fatalError(fatal);
- // Even if the application can continue ... we can't!
- throw fatal;
- }
- // We can safely report a few validity errors that
- // make layered SAX2 DTD validation more conformant
- void verror(String message)
- throws SAXException
- {
- SAXParseException err;
- err = new SAXParseException(message, this);
- errorHandler.error(err);
- }
- void warn(String message)
- throws SAXException
- {
- SAXParseException err;
- err = new SAXParseException(message, this);
- errorHandler.warning(err);
- }
- //
- // Implementation of org.xml.sax.Attributes.
- //
- /**
- * <b>SAX1 AttributeList, SAX2 Attributes</b> method
- * (don't invoke on parser);
- */
- public int getLength()
- {
- return attributesList.size();
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public String getURI(int index)
- {
- if (index < 0 || index >= attributesList.size())
- {
- return null;
- }
- return ((Attribute) attributesList.get(index)).nameSpace;
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public String getLocalName(int index)
- {
- if (index < 0 || index >= attributesList.size())
- {
- return null;
- }
- Attribute attr = (Attribute) attributesList.get(index);
- // FIXME attr.localName is sometimes null, why?
- if (namespaces && attr.localName == null)
- {
- // XXX fix this here for now
- int ci = attr.name.indexOf(':');
- attr.localName = (ci == -1) ? attr.name :
- attr.name.substring(ci + 1);
- }
- return (attr.localName == null) ? "" : attr.localName;
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public String getQName(int index)
- {
- if (index < 0 || index >= attributesList.size())
- {
- return null;
- }
- Attribute attr = (Attribute) attributesList.get(index);
- return (attr.name == null) ? "" : attr.name;
- }
- /**
- * <b>SAX1 AttributeList</b> method (don't invoke on parser);
- */
- public String getName(int index)
- {
- return getQName(index);
- }
- /**
- * <b>SAX1 AttributeList, SAX2 Attributes</b> method
- * (don't invoke on parser);
- */
- public String getType(int index)
- {
- if (index < 0 || index >= attributesList.size())
- {
- return null;
- }
- String type = parser.getAttributeType(elementName, getQName(index));
- if (type == null)
- {
- return "CDATA";
- }
- // ... use DeclHandler.attributeDecl to see enumerations
- if (type == "ENUMERATION")
- {
- return "NMTOKEN";
- }
- return type;
- }
- /**
- * <b>SAX1 AttributeList, SAX2 Attributes</b> method
- * (don't invoke on parser);
- */
- public String getValue(int index)
- {
- if (index < 0 || index >= attributesList.size())
- {
- return null;
- }
- return ((Attribute) attributesList.get(index)).value;
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public int getIndex(String uri, String local)
- {
- int length = getLength();
- for (int i = 0; i < length; i++)
- {
- if (!getURI(i).equals(uri))
- {
- continue;
- }
- if (getLocalName(i).equals(local))
- {
- return i;
- }
- }
- return -1;
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public int getIndex(String xmlName)
- {
- int length = getLength();
- for (int i = 0; i < length; i++)
- {
- if (getQName(i).equals(xmlName))
- {
- return i;
- }
- }
- return -1;
- }
- /**
- * <b>SAX2 Attributes</b> method (don't invoke on parser);
- */
- public String getType(String uri, String local)
- {
- int index = getIndex(uri, local);
- if (index < 0)
- {
- return null;
- }
- return getType(index);
- }
- /**
- * <b>SAX1 AttributeList, SAX2 Attributes</b> method
- * (don't invoke on parser);
- */
- public String getType(String xmlName)
- {
- int index = getIndex(xmlName);
- if (index < 0)
- {
- return null;
- }
- return getType(index);
- }
- /**
- * <b>SAX Attributes</b> method (don't invoke on parser);
- */
- public String getValue(String uri, String local)
- {
- int index = getIndex(uri, local);
- if (index < 0)
- {
- return null;
- }
- return getValue(index);
- }
- /**
- * <b>SAX1 AttributeList, SAX2 Attributes</b> method
- * (don't invoke on parser);
- */
- public String getValue(String xmlName)
- {
- int index = getIndex(xmlName);
- if (index < 0)
- {
- return null;
- }
- return getValue(index);
- }
- //
- // Implementation of org.xml.sax.ext.Attributes2
- //
- /** @return false unless the attribute was declared in the DTD.
- * @throws java.lang.ArrayIndexOutOfBoundsException
- * When the supplied index does not identify an attribute.
- */
- public boolean isDeclared(int index)
- {
- if (index < 0 || index >= attributeCount)
- {
- throw new ArrayIndexOutOfBoundsException();
- }
- String type = parser.getAttributeType(elementName, getQName(index));
- return (type != null);
- }
- /** @return false unless the attribute was declared in the DTD.
- * @throws java.lang.IllegalArgumentException
- * When the supplied names do not identify an attribute.
- */
- public boolean isDeclared(String qName)
- {
- int index = getIndex(qName);
- if (index < 0)
- {
- throw new IllegalArgumentException();
- }
- String type = parser.getAttributeType(elementName, qName);
- return (type != null);
- }
- /** @return false unless the attribute was declared in the DTD.
- * @throws java.lang.IllegalArgumentException
- * When the supplied names do not identify an attribute.
- */
- public boolean isDeclared(String uri, String localName)
- {
- int index = getIndex(uri, localName);
- return isDeclared(index);
- }
- /**
- * <b>SAX-ext Attributes2</b> method (don't invoke on parser);
- */
- public boolean isSpecified(int index)
- {
- return ((Attribute) attributesList.get(index)).specified;
- }
- /**
- * <b>SAX-ext Attributes2</b> method (don't invoke on parser);
- */
- public boolean isSpecified(String uri, String local)
- {
- int index = getIndex (uri, local);
- return isSpecified(index);
- }
- /**
- * <b>SAX-ext Attributes2</b> method (don't invoke on parser);
- */
- public boolean isSpecified(String xmlName)
- {
- int index = getIndex (xmlName);
- return isSpecified(index);
- }
- //
- // Implementation of org.xml.sax.Locator.
- //
- /**
- * <b>SAX Locator</b> method (don't invoke on parser);
- */
- public String getPublicId()
- {
- return null; // FIXME track public IDs too
- }
- /**
- * <b>SAX Locator</b> method (don't invoke on parser);
- */
- public String getSystemId()
- {
- if (entityStack.empty())
- {
- return null;
- }
- else
- {
- return (String) entityStack.peek();
- }
- }
- /**
- * <b>SAX Locator</b> method (don't invoke on parser);
- */
- public int getLineNumber()
- {
- return parser.getLineNumber();
- }
- /**
- * <b>SAX Locator</b> method (don't invoke on parser);
- */
- public int getColumnNumber()
- {
- return parser.getColumnNumber();
- }
- // adapter between SAX2 content handler and SAX1 document handler callbacks
- private static class Adapter
- implements ContentHandler
- {
- private DocumentHandler docHandler;
- Adapter(DocumentHandler dh)
- {
- docHandler = dh;
- }
- public void setDocumentLocator(Locator l)
- {
- docHandler.setDocumentLocator(l);
- }
- public void startDocument()
- throws SAXException
- {
- docHandler.startDocument();
- }
- public void processingInstruction(String target, String data)
- throws SAXException
- {
- docHandler.processingInstruction(target, data);
- }
- public void startPrefixMapping(String prefix, String uri)
- {
- /* ignored */
- }
- public void startElement(String namespace,
- String local,
- String name,
- Attributes attrs)
- throws SAXException
- {
- docHandler.startElement(name, (AttributeList) attrs);
- }
- public void characters(char[] buf, int offset, int len)
- throws SAXException
- {
- docHandler.characters(buf, offset, len);
- }
- public void ignorableWhitespace(char[] buf, int offset, int len)
- throws SAXException
- {
- docHandler.ignorableWhitespace(buf, offset, len);
- }
- public void skippedEntity(String name)
- {
- /* ignored */
- }
- public void endElement(String u, String l, String name)
- throws SAXException
- {
- docHandler.endElement(name);
- }
- public void endPrefixMapping(String prefix)
- {
- /* ignored */
- }
- public void endDocument()
- throws SAXException
- {
- docHandler.endDocument();
- }
- }
- private static class Attribute
- {
- String name;
- String value;
- String nameSpace;
- String localName;
- boolean specified;
- Attribute(String name, String value, boolean specified)
- {
- this.name = name;
- this.value = value;
- this.nameSpace = "";
- this.specified = specified;
- }
- }
- }
|